Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-winehq
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
wine
wine-winehq
Commits
857d4367
Commit
857d4367
authored
Aug 20, 2015
by
Sebastian Lackner
Committed by
Alexandre Julliard
Aug 20, 2015
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
ntdll: Implement FileRenameInformation support.
parent
6899eade
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
203 additions
and
19 deletions
+203
-19
file.c
dlls/ntdll/file.c
+43
-1
file.c
dlls/ntdll/tests/file.c
+0
-0
server_protocol.h
include/wine/server_protocol.h
+24
-6
fd.c
server/fd.c
+103
-2
protocol.def
server/protocol.def
+10
-2
request.h
server/request.h
+10
-5
trace.c
server/trace.c
+13
-3
No files found.
dlls/ntdll/file.c
View file @
857d4367
...
...
@@ -2770,7 +2770,7 @@ NTSTATUS WINAPI NtSetInformationFile(HANDLE handle, PIO_STATUS_BLOCK io,
{
FILE_DISPOSITION_INFORMATION
*
info
=
ptr
;
SERVER_START_REQ
(
set_fd_info
)
SERVER_START_REQ
(
set_fd_
disp_
info
)
{
req
->
handle
=
wine_server_obj_handle
(
handle
);
req
->
unlink
=
info
->
DoDeleteFile
;
...
...
@@ -2781,6 +2781,48 @@ NTSTATUS WINAPI NtSetInformationFile(HANDLE handle, PIO_STATUS_BLOCK io,
io
->
u
.
Status
=
STATUS_INVALID_PARAMETER_3
;
break
;
case
FileRenameInformation
:
if
(
len
>=
sizeof
(
FILE_RENAME_INFORMATION
))
{
FILE_RENAME_INFORMATION
*
info
=
ptr
;
UNICODE_STRING
name_str
;
OBJECT_ATTRIBUTES
attr
;
ANSI_STRING
unix_name
;
name_str
.
Buffer
=
info
->
FileName
;
name_str
.
Length
=
info
->
FileNameLength
;
name_str
.
MaximumLength
=
info
->
FileNameLength
+
sizeof
(
WCHAR
);
attr
.
Length
=
sizeof
(
attr
);
attr
.
ObjectName
=
&
name_str
;
attr
.
RootDirectory
=
info
->
RootDir
;
attr
.
Attributes
=
OBJ_CASE_INSENSITIVE
;
io
->
u
.
Status
=
nt_to_unix_file_name_attr
(
&
attr
,
&
unix_name
,
FILE_OPEN_IF
);
if
(
io
->
u
.
Status
!=
STATUS_SUCCESS
&&
io
->
u
.
Status
!=
STATUS_NO_SUCH_FILE
)
break
;
if
(
!
info
->
Replace
&&
io
->
u
.
Status
==
STATUS_SUCCESS
)
{
RtlFreeAnsiString
(
&
unix_name
);
io
->
u
.
Status
=
STATUS_OBJECT_NAME_COLLISION
;
break
;
}
SERVER_START_REQ
(
set_fd_name_info
)
{
req
->
handle
=
wine_server_obj_handle
(
handle
);
req
->
rootdir
=
wine_server_obj_handle
(
attr
.
RootDirectory
);
wine_server_add_data
(
req
,
unix_name
.
Buffer
,
unix_name
.
Length
);
io
->
u
.
Status
=
wine_server_call
(
req
);
}
SERVER_END_REQ
;
RtlFreeAnsiString
(
&
unix_name
);
}
else
io
->
u
.
Status
=
STATUS_INVALID_PARAMETER_3
;
break
;
default:
FIXME
(
"Unsupported class (%d)
\n
"
,
class
);
io
->
u
.
Status
=
STATUS_NOT_IMPLEMENTED
;
...
...
dlls/ntdll/tests/file.c
View file @
857d4367
This diff is collapsed.
Click to expand it.
include/wine/server_protocol.h
View file @
857d4367
...
...
@@ -5080,14 +5080,29 @@ struct add_fd_completion_reply
struct
set_fd_info_request
struct
set_fd_
disp_
info_request
{
struct
request_header
__header
;
obj_handle_t
handle
;
int
unlink
;
char
__pad_20
[
4
];
};
struct
set_fd_info_reply
struct
set_fd_disp_info_reply
{
struct
reply_header
__header
;
};
struct
set_fd_name_info_request
{
struct
request_header
__header
;
obj_handle_t
handle
;
obj_handle_t
rootdir
;
/* VARARG(filename,string); */
char
__pad_20
[
4
];
};
struct
set_fd_name_info_reply
{
struct
reply_header
__header
;
};
...
...
@@ -5566,7 +5581,8 @@ enum request
REQ_query_completion
,
REQ_set_completion_info
,
REQ_add_fd_completion
,
REQ_set_fd_info
,
REQ_set_fd_disp_info
,
REQ_set_fd_name_info
,
REQ_get_window_layered_info
,
REQ_set_window_layered_info
,
REQ_alloc_user_handle
,
...
...
@@ -5841,7 +5857,8 @@ union generic_request
struct
query_completion_request
query_completion_request
;
struct
set_completion_info_request
set_completion_info_request
;
struct
add_fd_completion_request
add_fd_completion_request
;
struct
set_fd_info_request
set_fd_info_request
;
struct
set_fd_disp_info_request
set_fd_disp_info_request
;
struct
set_fd_name_info_request
set_fd_name_info_request
;
struct
get_window_layered_info_request
get_window_layered_info_request
;
struct
set_window_layered_info_request
set_window_layered_info_request
;
struct
alloc_user_handle_request
alloc_user_handle_request
;
...
...
@@ -6114,7 +6131,8 @@ union generic_reply
struct
query_completion_reply
query_completion_reply
;
struct
set_completion_info_reply
set_completion_info_reply
;
struct
add_fd_completion_reply
add_fd_completion_reply
;
struct
set_fd_info_reply
set_fd_info_reply
;
struct
set_fd_disp_info_reply
set_fd_disp_info_reply
;
struct
set_fd_name_info_reply
set_fd_name_info_reply
;
struct
get_window_layered_info_reply
get_window_layered_info_reply
;
struct
set_window_layered_info_reply
set_window_layered_info_reply
;
struct
alloc_user_handle_reply
alloc_user_handle_reply
;
...
...
@@ -6131,6 +6149,6 @@ union generic_reply
struct
terminate_job_reply
terminate_job_reply
;
};
#define SERVER_PROTOCOL_VERSION 48
5
#define SERVER_PROTOCOL_VERSION 48
6
#endif
/* __WINE_WINE_SERVER_PROTOCOL_H */
server/fd.c
View file @
857d4367
...
...
@@ -2254,6 +2254,84 @@ static void set_fd_disposition( struct fd *fd, int unlink )
fd
->
closed
->
unlink
=
unlink
||
(
fd
->
options
&
FILE_DELETE_ON_CLOSE
);
}
/* set new name for the fd */
static
void
set_fd_name
(
struct
fd
*
fd
,
struct
fd
*
root
,
const
char
*
nameptr
,
data_size_t
len
)
{
struct
inode
*
inode
;
struct
stat
st
;
char
*
name
;
if
(
!
fd
->
inode
||
!
fd
->
unix_name
)
{
set_error
(
STATUS_OBJECT_TYPE_MISMATCH
);
return
;
}
if
(
!
len
||
((
nameptr
[
0
]
==
'/'
)
^
!
root
))
{
set_error
(
STATUS_OBJECT_PATH_SYNTAX_BAD
);
return
;
}
if
(
!
(
name
=
mem_alloc
(
len
+
1
)))
return
;
memcpy
(
name
,
nameptr
,
len
);
name
[
len
]
=
0
;
if
(
root
)
{
char
*
combined_name
=
dup_fd_name
(
root
,
name
);
if
(
!
combined_name
)
{
set_error
(
STATUS_NO_MEMORY
);
goto
failed
;
}
free
(
name
);
name
=
combined_name
;
}
if
(
!
stat
(
name
,
&
st
))
{
/* can't replace directories or special files */
if
(
!
S_ISREG
(
st
.
st_mode
))
{
set_error
(
STATUS_ACCESS_DENIED
);
goto
failed
;
}
/* can't replace an opened file */
if
((
inode
=
get_inode
(
st
.
st_dev
,
st
.
st_ino
,
-
1
)))
{
int
is_empty
=
list_empty
(
&
inode
->
open
);
release_object
(
inode
);
if
(
!
is_empty
)
{
set_error
(
STATUS_ACCESS_DENIED
);
goto
failed
;
}
}
/* rename() cannot replace files with directories */
if
(
fd
->
unix_fd
!=
-
1
&&
!
fstat
(
fd
->
unix_fd
,
&
st
)
&&
S_ISDIR
(
st
.
st_mode
)
&&
unlink
(
name
))
{
file_set_error
();
goto
failed
;
}
}
if
(
rename
(
fd
->
unix_name
,
name
))
{
file_set_error
();
goto
failed
;
}
free
(
fd
->
unix_name
);
fd
->
unix_name
=
name
;
fd
->
closed
->
unix_name
=
name
;
return
;
failed:
free
(
name
);
}
struct
completion
*
fd_get_completion
(
struct
fd
*
fd
,
apc_param_t
*
p_key
)
{
*
p_key
=
fd
->
comp_key
;
...
...
@@ -2450,8 +2528,8 @@ DECL_HANDLER(add_fd_completion)
}
}
/* set fd information */
DECL_HANDLER
(
set_fd_info
)
/* set fd
disposition
information */
DECL_HANDLER
(
set_fd_
disp_
info
)
{
struct
fd
*
fd
=
get_handle_fd_obj
(
current
->
process
,
req
->
handle
,
DELETE
);
if
(
fd
)
...
...
@@ -2460,3 +2538,26 @@ DECL_HANDLER(set_fd_info)
release_object
(
fd
);
}
}
/* set fd name information */
DECL_HANDLER
(
set_fd_name_info
)
{
struct
fd
*
fd
,
*
root_fd
=
NULL
;
if
(
req
->
rootdir
)
{
struct
dir
*
root
;
if
(
!
(
root
=
get_dir_obj
(
current
->
process
,
req
->
rootdir
,
0
)))
return
;
root_fd
=
get_obj_fd
(
(
struct
object
*
)
root
);
release_object
(
root
);
if
(
!
root_fd
)
return
;
}
if
((
fd
=
get_handle_fd_obj
(
current
->
process
,
req
->
handle
,
0
)))
{
set_fd_name
(
fd
,
root_fd
,
get_req_data
(),
get_req_data_size
()
);
release_object
(
fd
);
}
if
(
root_fd
)
release_object
(
root_fd
);
}
server/protocol.def
View file @
857d4367
...
...
@@ -3525,13 +3525,21 @@ enum coords_relative
@END
/* set fd information */
@REQ(set_fd_info)
/* set fd
disposition
information */
@REQ(set_fd_
disp_
info)
obj_handle_t handle; /* handle to a file or directory */
int unlink; /* whether to unlink file on close */
@END
/* set fd name information */
@REQ(set_fd_name_info)
obj_handle_t handle; /* handle to a file or directory */
obj_handle_t rootdir; /* root directory */
VARARG(filename,string); /* new file name */
@END
/* Retrieve layered info for a window */
@REQ(get_window_layered_info)
user_handle_t handle; /* handle to the window */
...
...
server/request.h
View file @
857d4367
...
...
@@ -359,7 +359,8 @@ DECL_HANDLER(remove_completion);
DECL_HANDLER
(
query_completion
);
DECL_HANDLER
(
set_completion_info
);
DECL_HANDLER
(
add_fd_completion
);
DECL_HANDLER
(
set_fd_info
);
DECL_HANDLER
(
set_fd_disp_info
);
DECL_HANDLER
(
set_fd_name_info
);
DECL_HANDLER
(
get_window_layered_info
);
DECL_HANDLER
(
set_window_layered_info
);
DECL_HANDLER
(
alloc_user_handle
);
...
...
@@ -633,7 +634,8 @@ static const req_handler req_handlers[REQ_NB_REQUESTS] =
(
req_handler
)
req_query_completion
,
(
req_handler
)
req_set_completion_info
,
(
req_handler
)
req_add_fd_completion
,
(
req_handler
)
req_set_fd_info
,
(
req_handler
)
req_set_fd_disp_info
,
(
req_handler
)
req_set_fd_name_info
,
(
req_handler
)
req_get_window_layered_info
,
(
req_handler
)
req_set_window_layered_info
,
(
req_handler
)
req_alloc_user_handle
,
...
...
@@ -2223,9 +2225,12 @@ C_ASSERT( FIELD_OFFSET(struct add_fd_completion_request, cvalue) == 16 );
C_ASSERT
(
FIELD_OFFSET
(
struct
add_fd_completion_request
,
information
)
==
24
);
C_ASSERT
(
FIELD_OFFSET
(
struct
add_fd_completion_request
,
status
)
==
32
);
C_ASSERT
(
sizeof
(
struct
add_fd_completion_request
)
==
40
);
C_ASSERT
(
FIELD_OFFSET
(
struct
set_fd_info_request
,
handle
)
==
12
);
C_ASSERT
(
FIELD_OFFSET
(
struct
set_fd_info_request
,
unlink
)
==
16
);
C_ASSERT
(
sizeof
(
struct
set_fd_info_request
)
==
24
);
C_ASSERT
(
FIELD_OFFSET
(
struct
set_fd_disp_info_request
,
handle
)
==
12
);
C_ASSERT
(
FIELD_OFFSET
(
struct
set_fd_disp_info_request
,
unlink
)
==
16
);
C_ASSERT
(
sizeof
(
struct
set_fd_disp_info_request
)
==
24
);
C_ASSERT
(
FIELD_OFFSET
(
struct
set_fd_name_info_request
,
handle
)
==
12
);
C_ASSERT
(
FIELD_OFFSET
(
struct
set_fd_name_info_request
,
rootdir
)
==
16
);
C_ASSERT
(
sizeof
(
struct
set_fd_name_info_request
)
==
24
);
C_ASSERT
(
FIELD_OFFSET
(
struct
get_window_layered_info_request
,
handle
)
==
12
);
C_ASSERT
(
sizeof
(
struct
get_window_layered_info_request
)
==
16
);
C_ASSERT
(
FIELD_OFFSET
(
struct
get_window_layered_info_reply
,
color_key
)
==
8
);
...
...
server/trace.c
View file @
857d4367
...
...
@@ -4136,12 +4136,19 @@ static void dump_add_fd_completion_request( const struct add_fd_completion_reque
fprintf
(
stderr
,
", status=%08x"
,
req
->
status
);
}
static
void
dump_set_fd_
info_request
(
const
struct
set_fd
_info_request
*
req
)
static
void
dump_set_fd_
disp_info_request
(
const
struct
set_fd_disp
_info_request
*
req
)
{
fprintf
(
stderr
,
" handle=%04x"
,
req
->
handle
);
fprintf
(
stderr
,
", unlink=%d"
,
req
->
unlink
);
}
static
void
dump_set_fd_name_info_request
(
const
struct
set_fd_name_info_request
*
req
)
{
fprintf
(
stderr
,
" handle=%04x"
,
req
->
handle
);
fprintf
(
stderr
,
", rootdir=%04x"
,
req
->
rootdir
);
dump_varargs_string
(
", filename="
,
cur_size
);
}
static
void
dump_get_window_layered_info_request
(
const
struct
get_window_layered_info_request
*
req
)
{
fprintf
(
stderr
,
" handle=%08x"
,
req
->
handle
);
...
...
@@ -4515,7 +4522,8 @@ static const dump_func req_dumpers[REQ_NB_REQUESTS] = {
(
dump_func
)
dump_query_completion_request
,
(
dump_func
)
dump_set_completion_info_request
,
(
dump_func
)
dump_add_fd_completion_request
,
(
dump_func
)
dump_set_fd_info_request
,
(
dump_func
)
dump_set_fd_disp_info_request
,
(
dump_func
)
dump_set_fd_name_info_request
,
(
dump_func
)
dump_get_window_layered_info_request
,
(
dump_func
)
dump_set_window_layered_info_request
,
(
dump_func
)
dump_alloc_user_handle_request
,
...
...
@@ -4787,6 +4795,7 @@ static const dump_func reply_dumpers[REQ_NB_REQUESTS] = {
NULL
,
NULL
,
NULL
,
NULL
,
(
dump_func
)
dump_get_window_layered_info_reply
,
NULL
,
(
dump_func
)
dump_alloc_user_handle_reply
,
...
...
@@ -5057,7 +5066,8 @@ static const char * const req_names[REQ_NB_REQUESTS] = {
"query_completion"
,
"set_completion_info"
,
"add_fd_completion"
,
"set_fd_info"
,
"set_fd_disp_info"
,
"set_fd_name_info"
,
"get_window_layered_info"
,
"set_window_layered_info"
,
"alloc_user_handle"
,
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment