Skip to content

Commit 6ddea07

Browse files
CanadaHonktargos
authored andcommitted
fs: improve error perf of sync chmod+fchmod
PR-URL: #49859 Refs: nodejs/performance#106 Reviewed-By: Yagiz Nizipli <[email protected]> Reviewed-By: Darshan Sen <[email protected]> Reviewed-By: Joyee Cheung <[email protected]>
1 parent 8413670 commit 6ddea07

File tree

5 files changed

+121
-24
lines changed

5 files changed

+121
-24
lines changed

benchmark/fs/bench-chmodSync.js

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
'use strict';
2+
3+
const common = require('../common');
4+
const fs = require('fs');
5+
const tmpdir = require('../../test/common/tmpdir');
6+
tmpdir.refresh();
7+
8+
const bench = common.createBenchmark(main, {
9+
type: ['existing', 'non-existing'],
10+
n: [1e3],
11+
});
12+
13+
function main({ n, type }) {
14+
switch (type) {
15+
case 'existing': {
16+
for (let i = 0; i < n; i++) {
17+
fs.writeFileSync(tmpdir.resolve(`chmodsync-bench-file-${i}`), 'bench');
18+
}
19+
20+
bench.start();
21+
for (let i = 0; i < n; i++) {
22+
fs.chmodSync(tmpdir.resolve(`chmodsync-bench-file-${i}`), 0o666);
23+
}
24+
bench.end(n);
25+
break;
26+
}
27+
case 'non-existing':
28+
bench.start();
29+
for (let i = 0; i < n; i++) {
30+
try {
31+
fs.chmodSync(tmpdir.resolve(`chmod-non-existing-file-${i}`), 0o666);
32+
} catch {
33+
// do nothing
34+
}
35+
}
36+
bench.end(n);
37+
break;
38+
default:
39+
new Error('Invalid type');
40+
}
41+
}

benchmark/fs/bench-fchmodSync.js

+60
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
'use strict';
2+
3+
const common = require('../common');
4+
const fs = require('fs');
5+
const tmpdir = require('../../test/common/tmpdir');
6+
tmpdir.refresh();
7+
8+
const bench = common.createBenchmark(main, {
9+
type: ['existing', 'non-existing'],
10+
n: [1e3],
11+
});
12+
13+
function main({ n, type }) {
14+
let files;
15+
16+
switch (type) {
17+
case 'existing':
18+
files = [];
19+
20+
// Populate tmpdir with mock files
21+
for (let i = 0; i < n; i++) {
22+
const path = tmpdir.resolve(`fchmodsync-bench-file-${i}`);
23+
fs.writeFileSync(path, 'bench');
24+
files.push(path);
25+
}
26+
break;
27+
case 'non-existing':
28+
files = new Array(n).fill(tmpdir.resolve(`.non-existing-file-${Date.now()}`));
29+
break;
30+
default:
31+
new Error('Invalid type');
32+
}
33+
34+
const fds = files.map((x) => {
35+
// Try to open, if not return likely invalid fd (1 << 30)
36+
try {
37+
return fs.openSync(x, 'r');
38+
} catch {
39+
return 1 << 30;
40+
}
41+
});
42+
43+
bench.start();
44+
for (let i = 0; i < n; i++) {
45+
try {
46+
fs.fchmodSync(fds[i], 0o666);
47+
} catch {
48+
// do nothing
49+
}
50+
}
51+
bench.end(n);
52+
53+
for (const x of fds) {
54+
try {
55+
fs.closeSync(x);
56+
} catch {
57+
// do nothing
58+
}
59+
}
60+
}

lib/fs.js

+8-8
Original file line numberDiff line numberDiff line change
@@ -1901,11 +1901,10 @@ function fchmod(fd, mode, callback) {
19011901
* @returns {void}
19021902
*/
19031903
function fchmodSync(fd, mode) {
1904-
fd = getValidatedFd(fd);
1905-
mode = parseFileMode(mode, 'mode');
1906-
const ctx = {};
1907-
binding.fchmod(fd, mode, undefined, ctx);
1908-
handleErrorFromBinding(ctx);
1904+
binding.fchmod(
1905+
getValidatedFd(fd),
1906+
parseFileMode(mode, 'mode'),
1907+
);
19091908
}
19101909

19111910
/**
@@ -1980,9 +1979,10 @@ function chmodSync(path, mode) {
19801979
path = getValidatedPath(path);
19811980
mode = parseFileMode(mode, 'mode');
19821981

1983-
const ctx = { path };
1984-
binding.chmod(pathModule.toNamespacedPath(path), mode, undefined, ctx);
1985-
handleErrorFromBinding(ctx);
1982+
binding.chmod(
1983+
pathModule.toNamespacedPath(path),
1984+
mode,
1985+
);
19861986
}
19871987

19881988
/**

src/node_file.cc

+10-14
Original file line numberDiff line numberDiff line change
@@ -2552,18 +2552,16 @@ static void Chmod(const FunctionCallbackInfo<Value>& args) {
25522552
CHECK(args[1]->IsInt32());
25532553
int mode = args[1].As<Int32>()->Value();
25542554

2555-
FSReqBase* req_wrap_async = GetReqWrap(args, 2);
2556-
if (req_wrap_async != nullptr) { // chmod(path, mode, req)
2555+
if (argc > 2) { // chmod(path, mode, req)
2556+
FSReqBase* req_wrap_async = GetReqWrap(args, 2);
25572557
FS_ASYNC_TRACE_BEGIN1(
25582558
UV_FS_CHMOD, req_wrap_async, "path", TRACE_STR_COPY(*path))
25592559
AsyncCall(env, req_wrap_async, args, "chmod", UTF8, AfterNoArgs,
25602560
uv_fs_chmod, *path, mode);
2561-
} else { // chmod(path, mode, undefined, ctx)
2562-
CHECK_EQ(argc, 4);
2563-
FSReqWrapSync req_wrap_sync;
2561+
} else { // chmod(path, mode)
2562+
FSReqWrapSync req_wrap_sync("chmod", *path);
25642563
FS_SYNC_TRACE_BEGIN(chmod);
2565-
SyncCall(env, args[3], &req_wrap_sync, "chmod",
2566-
uv_fs_chmod, *path, mode);
2564+
SyncCallAndThrowOnError(env, &req_wrap_sync, uv_fs_chmod, *path, mode);
25672565
FS_SYNC_TRACE_END(chmod);
25682566
}
25692567
}
@@ -2584,17 +2582,15 @@ static void FChmod(const FunctionCallbackInfo<Value>& args) {
25842582
CHECK(args[1]->IsInt32());
25852583
const int mode = args[1].As<Int32>()->Value();
25862584

2587-
FSReqBase* req_wrap_async = GetReqWrap(args, 2);
2588-
if (req_wrap_async != nullptr) { // fchmod(fd, mode, req)
2585+
if (argc > 2) { // fchmod(fd, mode, req)
2586+
FSReqBase* req_wrap_async = GetReqWrap(args, 2);
25892587
FS_ASYNC_TRACE_BEGIN0(UV_FS_FCHMOD, req_wrap_async)
25902588
AsyncCall(env, req_wrap_async, args, "fchmod", UTF8, AfterNoArgs,
25912589
uv_fs_fchmod, fd, mode);
2592-
} else { // fchmod(fd, mode, undefined, ctx)
2593-
CHECK_EQ(argc, 4);
2594-
FSReqWrapSync req_wrap_sync;
2590+
} else { // fchmod(fd, mode)
2591+
FSReqWrapSync req_wrap_sync("fchmod");
25952592
FS_SYNC_TRACE_BEGIN(fchmod);
2596-
SyncCall(env, args[3], &req_wrap_sync, "fchmod",
2597-
uv_fs_fchmod, fd, mode);
2593+
SyncCallAndThrowOnError(env, &req_wrap_sync, uv_fs_fchmod, fd, mode);
25982594
FS_SYNC_TRACE_END(fchmod);
25992595
}
26002596
}

typings/internalBinding/fs.d.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,7 @@ declare namespace InternalFSBinding {
6161
function access(path: StringOrBuffer, mode: number, usePromises: typeof kUsePromises): Promise<void>;
6262

6363
function chmod(path: string, mode: number, req: FSReqCallback): void;
64-
function chmod(path: string, mode: number, req: undefined, ctx: FSSyncContext): void;
64+
function chmod(path: string, mode: number): void;
6565
function chmod(path: string, mode: number, usePromises: typeof kUsePromises): Promise<void>;
6666

6767
function chown(path: string, uid: number, gid: number, req: FSReqCallback): void;
@@ -76,7 +76,7 @@ declare namespace InternalFSBinding {
7676
function copyFile(src: StringOrBuffer, dest: StringOrBuffer, mode: number, usePromises: typeof kUsePromises): Promise<void>;
7777

7878
function fchmod(fd: number, mode: number, req: FSReqCallback): void;
79-
function fchmod(fd: number, mode: number, req: undefined, ctx: FSSyncContext): void;
79+
function fchmod(fd: number, mode: number): void;
8080
function fchmod(fd: number, mode: number, usePromises: typeof kUsePromises): Promise<void>;
8181

8282
function fchown(fd: number, uid: number, gid: number, req: FSReqCallback): void;

0 commit comments

Comments
 (0)