2 * Object management functions
4 * Copyright 1999, 2000 Juergen Schmied
5 * Copyright 2005 Vitaliy Margolen
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
35 #define WIN32_NO_STATUS
36 #include "wine/debug.h"
39 #include "ntdll_misc.h"
40 #include "wine/server.h"
42 WINE_DEFAULT_DEBUG_CHANNEL(ntdll
);
46 * Generic object functions
49 /******************************************************************************
50 * NtQueryObject [NTDLL.@]
51 * ZwQueryObject [NTDLL.@]
53 NTSTATUS WINAPI
NtQueryObject(IN HANDLE handle
,
54 IN OBJECT_INFORMATION_CLASS info_class
,
55 OUT PVOID ptr
, IN ULONG len
, OUT PULONG used_len
)
59 TRACE("(%p,0x%08x,%p,0x%08x,%p): stub\n",
60 handle
, info_class
, ptr
, len
, used_len
);
62 if (used_len
) *used_len
= 0;
66 case ObjectBasicInformation
:
68 POBJECT_BASIC_INFORMATION p
= ptr
;
70 if (len
< sizeof(*p
)) return STATUS_INVALID_BUFFER_SIZE
;
72 SERVER_START_REQ( get_object_info
)
74 req
->handle
= wine_server_obj_handle( handle
);
75 status
= wine_server_call( req
);
76 if (status
== STATUS_SUCCESS
)
78 memset( p
, 0, sizeof(*p
) );
79 p
->GrantedAccess
= reply
->access
;
80 p
->PointerCount
= reply
->ref_count
;
81 p
->HandleCount
= 1; /* at least one */
82 if (used_len
) *used_len
= sizeof(*p
);
88 case ObjectNameInformation
:
90 OBJECT_NAME_INFORMATION
* p
= ptr
;
91 ANSI_STRING unix_name
;
93 /* first try as a file object */
95 if (!(status
= server_get_unix_name( handle
, &unix_name
)))
97 UNICODE_STRING nt_name
;
99 if (!(status
= wine_unix_to_nt_file_name( &unix_name
, &nt_name
)))
101 if (len
< sizeof(*p
))
102 status
= STATUS_INFO_LENGTH_MISMATCH
;
103 else if (len
< sizeof(*p
) + nt_name
.MaximumLength
)
104 status
= STATUS_BUFFER_OVERFLOW
;
107 p
->Name
.Buffer
= (WCHAR
*)(p
+ 1);
108 p
->Name
.Length
= nt_name
.Length
;
109 p
->Name
.MaximumLength
= nt_name
.MaximumLength
;
110 memcpy( p
->Name
.Buffer
, nt_name
.Buffer
, nt_name
.MaximumLength
);
112 if (used_len
) *used_len
= sizeof(*p
) + nt_name
.MaximumLength
;
113 RtlFreeUnicodeString( &nt_name
);
115 RtlFreeAnsiString( &unix_name
);
118 else if (status
!= STATUS_OBJECT_TYPE_MISMATCH
) break;
120 /* not a file, treat as a generic object */
122 SERVER_START_REQ( get_object_info
)
124 req
->handle
= wine_server_obj_handle( handle
);
125 if (len
> sizeof(*p
)) wine_server_set_reply( req
, p
+ 1, len
- sizeof(*p
) );
126 status
= wine_server_call( req
);
127 if (status
== STATUS_SUCCESS
)
129 if (!reply
->total
) /* no name */
131 if (sizeof(*p
) > len
) status
= STATUS_INFO_LENGTH_MISMATCH
;
132 else memset( p
, 0, sizeof(*p
) );
133 if (used_len
) *used_len
= sizeof(*p
);
135 else if (sizeof(*p
) + reply
->total
+ sizeof(WCHAR
) > len
)
137 if (used_len
) *used_len
= sizeof(*p
) + reply
->total
+ sizeof(WCHAR
);
138 status
= STATUS_INFO_LENGTH_MISMATCH
;
142 ULONG res
= wine_server_reply_size( reply
);
143 p
->Name
.Buffer
= (WCHAR
*)(p
+ 1);
144 p
->Name
.Length
= res
;
145 p
->Name
.MaximumLength
= res
+ sizeof(WCHAR
);
146 p
->Name
.Buffer
[res
/ sizeof(WCHAR
)] = 0;
147 if (used_len
) *used_len
= sizeof(*p
) + p
->Name
.MaximumLength
;
154 case ObjectDataInformation
:
156 OBJECT_DATA_INFORMATION
* p
= ptr
;
158 if (len
< sizeof(*p
)) return STATUS_INVALID_BUFFER_SIZE
;
160 SERVER_START_REQ( set_handle_info
)
162 req
->handle
= wine_server_obj_handle( handle
);
165 status
= wine_server_call( req
);
166 if (status
== STATUS_SUCCESS
)
168 p
->InheritHandle
= (reply
->old_flags
& HANDLE_FLAG_INHERIT
) ? TRUE
: FALSE
;
169 p
->ProtectFromClose
= (reply
->old_flags
& HANDLE_FLAG_PROTECT_FROM_CLOSE
) ? TRUE
: FALSE
;
170 if (used_len
) *used_len
= sizeof(*p
);
177 FIXME("Unsupported information class %u\n", info_class
);
178 status
= STATUS_NOT_IMPLEMENTED
;
184 /******************************************************************
185 * NtSetInformationObject [NTDLL.@]
186 * ZwSetInformationObject [NTDLL.@]
189 NTSTATUS WINAPI
NtSetInformationObject(IN HANDLE handle
,
190 IN OBJECT_INFORMATION_CLASS info_class
,
191 IN PVOID ptr
, IN ULONG len
)
195 TRACE("(%p,0x%08x,%p,0x%08x): stub\n",
196 handle
, info_class
, ptr
, len
);
200 case ObjectDataInformation
:
202 OBJECT_DATA_INFORMATION
* p
= ptr
;
204 if (len
< sizeof(*p
)) return STATUS_INVALID_BUFFER_SIZE
;
206 SERVER_START_REQ( set_handle_info
)
208 req
->handle
= wine_server_obj_handle( handle
);
210 req
->mask
= HANDLE_FLAG_INHERIT
| HANDLE_FLAG_PROTECT_FROM_CLOSE
;
211 if (p
->InheritHandle
) req
->flags
|= HANDLE_FLAG_INHERIT
;
212 if (p
->ProtectFromClose
) req
->flags
|= HANDLE_FLAG_PROTECT_FROM_CLOSE
;
213 status
= wine_server_call( req
);
219 FIXME("Unsupported information class %u\n", info_class
);
220 status
= STATUS_NOT_IMPLEMENTED
;
226 /******************************************************************************
227 * NtQuerySecurityObject [NTDLL.@]
229 * An ntdll analogue to GetKernelObjectSecurity().
233 NtQuerySecurityObject(
235 IN SECURITY_INFORMATION RequestedInformation
,
236 OUT PSECURITY_DESCRIPTOR pSecurityDescriptor
,
238 OUT PULONG ResultLength
)
240 PISECURITY_DESCRIPTOR_RELATIVE psd
= pSecurityDescriptor
;
242 unsigned int buffer_size
= 512;
243 BOOLEAN need_more_memory
;
245 TRACE("(%p,0x%08x,%p,0x%08x,%p)\n",
246 Object
, RequestedInformation
, pSecurityDescriptor
, Length
, ResultLength
);
250 char *buffer
= RtlAllocateHeap(GetProcessHeap(), 0, buffer_size
);
252 return STATUS_NO_MEMORY
;
254 need_more_memory
= FALSE
;
256 SERVER_START_REQ( get_security_object
)
258 req
->handle
= wine_server_obj_handle( Object
);
259 req
->security_info
= RequestedInformation
;
260 wine_server_set_reply( req
, buffer
, buffer_size
);
261 status
= wine_server_call( req
);
262 if (status
== STATUS_SUCCESS
)
264 struct security_descriptor
*sd
= (struct security_descriptor
*)buffer
;
267 *ResultLength
= sizeof(SECURITY_DESCRIPTOR_RELATIVE
) +
268 sd
->owner_len
+ sd
->group_len
+ sd
->sacl_len
+ sd
->dacl_len
;
269 if (Length
>= *ResultLength
)
271 psd
->Revision
= SECURITY_DESCRIPTOR_REVISION
;
273 psd
->Control
= sd
->control
| SE_SELF_RELATIVE
;
274 psd
->Owner
= sd
->owner_len
? sizeof(SECURITY_DESCRIPTOR_RELATIVE
) : 0;
275 psd
->Group
= sd
->group_len
? sizeof(SECURITY_DESCRIPTOR_RELATIVE
) + sd
->owner_len
: 0;
276 psd
->Sacl
= sd
->sacl_len
? sizeof(SECURITY_DESCRIPTOR_RELATIVE
) + sd
->owner_len
+ sd
->group_len
: 0;
277 psd
->Dacl
= sd
->dacl_len
? sizeof(SECURITY_DESCRIPTOR_RELATIVE
) + sd
->owner_len
+ sd
->group_len
+ sd
->sacl_len
: 0;
278 /* owner, group, sacl and dacl are the same type as in the server
279 * and in the same order so we copy the memory in one block */
280 memcpy((char *)pSecurityDescriptor
+ sizeof(SECURITY_DESCRIPTOR_RELATIVE
),
281 buffer
+ sizeof(struct security_descriptor
),
282 sd
->owner_len
+ sd
->group_len
+ sd
->sacl_len
+ sd
->dacl_len
);
285 status
= STATUS_BUFFER_TOO_SMALL
;
289 *ResultLength
= sizeof(SECURITY_DESCRIPTOR_RELATIVE
);
290 if (Length
>= *ResultLength
)
292 memset(psd
, 0, sizeof(*psd
));
293 psd
->Revision
= SECURITY_DESCRIPTOR_REVISION
;
294 psd
->Control
= SE_SELF_RELATIVE
;
297 status
= STATUS_BUFFER_TOO_SMALL
;
300 else if (status
== STATUS_BUFFER_TOO_SMALL
)
302 buffer_size
= reply
->sd_len
;
303 need_more_memory
= TRUE
;
307 RtlFreeHeap(GetProcessHeap(), 0, buffer
);
308 } while (need_more_memory
);
314 /******************************************************************************
315 * NtDuplicateObject [NTDLL.@]
316 * ZwDuplicateObject [NTDLL.@]
318 NTSTATUS WINAPI
NtDuplicateObject( HANDLE source_process
, HANDLE source
,
319 HANDLE dest_process
, PHANDLE dest
,
320 ACCESS_MASK access
, ULONG attributes
, ULONG options
)
323 SERVER_START_REQ( dup_handle
)
325 req
->src_process
= wine_server_obj_handle( source_process
);
326 req
->src_handle
= wine_server_obj_handle( source
);
327 req
->dst_process
= wine_server_obj_handle( dest_process
);
328 req
->access
= access
;
329 req
->attributes
= attributes
;
330 req
->options
= options
;
332 if (!(ret
= wine_server_call( req
)))
334 if (dest
) *dest
= wine_server_ptr_handle( reply
->handle
);
339 int fd
= server_remove_fd_from_cache( source
);
340 if (fd
!= -1) close( fd
);
343 else if (options
& DUPLICATE_CLOSE_SOURCE
)
344 WARN( "failed to close handle %p in process %p\n", source
, source_process
);
351 /**************************************************************************
354 * Close a handle reference to an object.
357 * Handle [I] handle to close
360 * Success: ERROR_SUCCESS.
361 * Failure: An NTSTATUS error code.
363 NTSTATUS WINAPI
NtClose( HANDLE Handle
)
366 int fd
= server_remove_fd_from_cache( Handle
);
368 SERVER_START_REQ( close_handle
)
370 req
->handle
= wine_server_obj_handle( Handle
);
371 ret
= wine_server_call( req
);
374 if (fd
!= -1) close( fd
);
379 * Directory functions
382 /**************************************************************************
383 * NtOpenDirectoryObject [NTDLL.@]
384 * ZwOpenDirectoryObject [NTDLL.@]
386 * Open a namespace directory object.
389 * DirectoryHandle [O] Destination for the new directory handle
390 * DesiredAccess [I] Desired access to the directory
391 * ObjectAttributes [I] Structure describing the directory
394 * Success: ERROR_SUCCESS.
395 * Failure: An NTSTATUS error code.
397 NTSTATUS WINAPI
NtOpenDirectoryObject(PHANDLE DirectoryHandle
, ACCESS_MASK DesiredAccess
,
398 POBJECT_ATTRIBUTES ObjectAttributes
)
401 TRACE("(%p,0x%08x,%s)\n", DirectoryHandle
, DesiredAccess
, debugstr_ObjectAttributes(ObjectAttributes
));
403 if (!DirectoryHandle
) return STATUS_ACCESS_VIOLATION
;
404 if (!ObjectAttributes
) return STATUS_INVALID_PARAMETER
;
405 /* Have to test it here because server won't know difference between
406 * ObjectName == NULL and ObjectName == "" */
407 if (!ObjectAttributes
->ObjectName
)
409 if (ObjectAttributes
->RootDirectory
)
410 return STATUS_OBJECT_NAME_INVALID
;
412 return STATUS_OBJECT_PATH_SYNTAX_BAD
;
415 SERVER_START_REQ(open_directory
)
417 req
->access
= DesiredAccess
;
418 req
->attributes
= ObjectAttributes
->Attributes
;
419 req
->rootdir
= wine_server_obj_handle( ObjectAttributes
->RootDirectory
);
420 if (ObjectAttributes
->ObjectName
)
421 wine_server_add_data(req
, ObjectAttributes
->ObjectName
->Buffer
,
422 ObjectAttributes
->ObjectName
->Length
);
423 ret
= wine_server_call( req
);
424 *DirectoryHandle
= wine_server_ptr_handle( reply
->handle
);
430 /******************************************************************************
431 * NtCreateDirectoryObject [NTDLL.@]
432 * ZwCreateDirectoryObject [NTDLL.@]
434 * Create a namespace directory object.
437 * DirectoryHandle [O] Destination for the new directory handle
438 * DesiredAccess [I] Desired access to the directory
439 * ObjectAttributes [I] Structure describing the directory
442 * Success: ERROR_SUCCESS.
443 * Failure: An NTSTATUS error code.
445 NTSTATUS WINAPI
NtCreateDirectoryObject(PHANDLE DirectoryHandle
, ACCESS_MASK DesiredAccess
,
446 POBJECT_ATTRIBUTES ObjectAttributes
)
449 TRACE("(%p,0x%08x,%s)\n", DirectoryHandle
, DesiredAccess
, debugstr_ObjectAttributes(ObjectAttributes
));
451 if (!DirectoryHandle
) return STATUS_ACCESS_VIOLATION
;
453 SERVER_START_REQ(create_directory
)
455 req
->access
= DesiredAccess
;
456 req
->attributes
= ObjectAttributes
? ObjectAttributes
->Attributes
: 0;
457 req
->rootdir
= wine_server_obj_handle( ObjectAttributes
? ObjectAttributes
->RootDirectory
: 0 );
458 if (ObjectAttributes
&& ObjectAttributes
->ObjectName
)
459 wine_server_add_data(req
, ObjectAttributes
->ObjectName
->Buffer
,
460 ObjectAttributes
->ObjectName
->Length
);
461 ret
= wine_server_call( req
);
462 *DirectoryHandle
= wine_server_ptr_handle( reply
->handle
);
468 /******************************************************************************
469 * NtQueryDirectoryObject [NTDLL.@]
470 * ZwQueryDirectoryObject [NTDLL.@]
472 * Read information from a namespace directory.
475 * handle [I] Handle to a directory object
476 * buffer [O] Buffer to hold the read data
477 * size [I] Size of the buffer in bytes
478 * single_entry [I] If TRUE, return a single entry, if FALSE, return as many as fit in the buffer
479 * restart [I] If TRUE, start scanning from the start, if FALSE, scan from Context
480 * context [I/O] Indicates what point of the directory the scan is at
481 * ret_size [O] Caller supplied storage for the number of bytes written (or NULL)
484 * Success: ERROR_SUCCESS.
485 * Failure: An NTSTATUS error code.
487 NTSTATUS WINAPI
NtQueryDirectoryObject(HANDLE handle
, PDIRECTORY_BASIC_INFORMATION buffer
,
488 ULONG size
, BOOLEAN single_entry
, BOOLEAN restart
,
489 PULONG context
, PULONG ret_size
)
493 if (restart
) *context
= 0;
497 if (size
<= sizeof(*buffer
) + 2*sizeof(WCHAR
)) return STATUS_BUFFER_OVERFLOW
;
499 SERVER_START_REQ( get_directory_entry
)
501 req
->handle
= wine_server_obj_handle( handle
);
502 req
->index
= *context
;
503 wine_server_set_reply( req
, buffer
+ 1, size
- sizeof(*buffer
) - 2*sizeof(WCHAR
) );
504 if (!(ret
= wine_server_call( req
)))
506 buffer
->ObjectName
.Buffer
= (WCHAR
*)(buffer
+ 1);
507 buffer
->ObjectName
.Length
= reply
->name_len
;
508 buffer
->ObjectName
.MaximumLength
= reply
->name_len
+ sizeof(WCHAR
);
509 buffer
->ObjectTypeName
.Buffer
= (WCHAR
*)(buffer
+ 1) + reply
->name_len
/sizeof(WCHAR
) + 1;
510 buffer
->ObjectTypeName
.Length
= wine_server_reply_size( reply
) - reply
->name_len
;
511 buffer
->ObjectTypeName
.MaximumLength
= buffer
->ObjectTypeName
.Length
+ sizeof(WCHAR
);
512 /* make room for the terminating null */
513 memmove( buffer
->ObjectTypeName
.Buffer
, buffer
->ObjectTypeName
.Buffer
- 1,
514 buffer
->ObjectTypeName
.Length
);
515 buffer
->ObjectName
.Buffer
[buffer
->ObjectName
.Length
/sizeof(WCHAR
)] = 0;
516 buffer
->ObjectTypeName
.Buffer
[buffer
->ObjectTypeName
.Length
/sizeof(WCHAR
)] = 0;
522 *ret_size
= buffer
->ObjectName
.MaximumLength
+ buffer
->ObjectTypeName
.MaximumLength
+ sizeof(*buffer
);
526 FIXME("multiple entries not implemented\n");
527 ret
= STATUS_NOT_IMPLEMENTED
;
537 /******************************************************************************
538 * NtOpenSymbolicLinkObject [NTDLL.@]
539 * ZwOpenSymbolicLinkObject [NTDLL.@]
541 * Open a namespace symbolic link object.
544 * LinkHandle [O] Destination for the new symbolic link handle
545 * DesiredAccess [I] Desired access to the symbolic link
546 * ObjectAttributes [I] Structure describing the symbolic link
549 * Success: ERROR_SUCCESS.
550 * Failure: An NTSTATUS error code.
552 NTSTATUS WINAPI
NtOpenSymbolicLinkObject(OUT PHANDLE LinkHandle
, IN ACCESS_MASK DesiredAccess
,
553 IN POBJECT_ATTRIBUTES ObjectAttributes
)
556 TRACE("(%p,0x%08x,%s)\n",LinkHandle
, DesiredAccess
, debugstr_ObjectAttributes(ObjectAttributes
));
558 if (!LinkHandle
) return STATUS_ACCESS_VIOLATION
;
559 if (!ObjectAttributes
) return STATUS_INVALID_PARAMETER
;
560 /* Have to test it here because server won't know difference between
561 * ObjectName == NULL and ObjectName == "" */
562 if (!ObjectAttributes
->ObjectName
)
564 if (ObjectAttributes
->RootDirectory
)
565 return STATUS_OBJECT_NAME_INVALID
;
567 return STATUS_OBJECT_PATH_SYNTAX_BAD
;
570 SERVER_START_REQ(open_symlink
)
572 req
->access
= DesiredAccess
;
573 req
->attributes
= ObjectAttributes
->Attributes
;
574 req
->rootdir
= wine_server_obj_handle( ObjectAttributes
->RootDirectory
);
575 if (ObjectAttributes
->ObjectName
)
576 wine_server_add_data(req
, ObjectAttributes
->ObjectName
->Buffer
,
577 ObjectAttributes
->ObjectName
->Length
);
578 ret
= wine_server_call( req
);
579 *LinkHandle
= wine_server_ptr_handle( reply
->handle
);
585 /******************************************************************************
586 * NtCreateSymbolicLinkObject [NTDLL.@]
587 * ZwCreateSymbolicLinkObject [NTDLL.@]
589 * Open a namespace symbolic link object.
592 * SymbolicLinkHandle [O] Destination for the new symbolic link handle
593 * DesiredAccess [I] Desired access to the symbolic link
594 * ObjectAttributes [I] Structure describing the symbolic link
595 * TargetName [I] Name of the target symbolic link points to
598 * Success: ERROR_SUCCESS.
599 * Failure: An NTSTATUS error code.
601 NTSTATUS WINAPI
NtCreateSymbolicLinkObject(OUT PHANDLE SymbolicLinkHandle
,IN ACCESS_MASK DesiredAccess
,
602 IN POBJECT_ATTRIBUTES ObjectAttributes
,
603 IN PUNICODE_STRING TargetName
)
607 if (!SymbolicLinkHandle
|| !TargetName
) return STATUS_ACCESS_VIOLATION
;
608 if (!TargetName
->Buffer
) return STATUS_INVALID_PARAMETER
;
610 TRACE("(%p,0x%08x,%s -> %s)\n", SymbolicLinkHandle
, DesiredAccess
,
611 debugstr_ObjectAttributes(ObjectAttributes
), debugstr_us(TargetName
));
613 SERVER_START_REQ(create_symlink
)
615 req
->access
= DesiredAccess
;
616 req
->attributes
= ObjectAttributes
? ObjectAttributes
->Attributes
: 0;
617 req
->rootdir
= wine_server_obj_handle( ObjectAttributes
? ObjectAttributes
->RootDirectory
: 0 );
618 if (ObjectAttributes
&& ObjectAttributes
->ObjectName
)
620 req
->name_len
= ObjectAttributes
->ObjectName
->Length
;
621 wine_server_add_data(req
, ObjectAttributes
->ObjectName
->Buffer
,
622 ObjectAttributes
->ObjectName
->Length
);
626 wine_server_add_data(req
, TargetName
->Buffer
, TargetName
->Length
);
627 ret
= wine_server_call( req
);
628 *SymbolicLinkHandle
= wine_server_ptr_handle( reply
->handle
);
634 /******************************************************************************
635 * NtQuerySymbolicLinkObject [NTDLL.@]
636 * ZwQuerySymbolicLinkObject [NTDLL.@]
638 * Query a namespace symbolic link object target name.
641 * handle [I] Handle to a symbolic link object
642 * target [O] Destination for the symbolic link target
643 * length [O] Size of returned data
646 * Success: ERROR_SUCCESS.
647 * Failure: An NTSTATUS error code.
649 NTSTATUS WINAPI
NtQuerySymbolicLinkObject( HANDLE handle
, PUNICODE_STRING target
, PULONG length
)
653 TRACE("(%p,%p,%p)\n", handle
, target
, length
);
655 if (!target
) return STATUS_ACCESS_VIOLATION
;
657 SERVER_START_REQ(query_symlink
)
659 req
->handle
= wine_server_obj_handle( handle
);
660 if (target
->MaximumLength
>= sizeof(WCHAR
))
661 wine_server_set_reply( req
, target
->Buffer
, target
->MaximumLength
- sizeof(WCHAR
) );
662 if (!(ret
= wine_server_call( req
)))
664 target
->Length
= wine_server_reply_size(reply
);
665 target
->Buffer
[target
->Length
/ sizeof(WCHAR
)] = 0;
666 if (length
) *length
= reply
->total
+ sizeof(WCHAR
);
668 else if (length
&& ret
== STATUS_BUFFER_TOO_SMALL
) *length
= reply
->total
+ sizeof(WCHAR
);
674 /******************************************************************************
675 * NtAllocateUuids [NTDLL.@]
677 NTSTATUS WINAPI
NtAllocateUuids(
678 PULARGE_INTEGER Time
,
682 FIXME("(%p,%p,%p), stub.\n", Time
, Range
, Sequence
);
686 /**************************************************************************
687 * NtMakeTemporaryObject [NTDLL.@]
688 * ZwMakeTemporaryObject [NTDLL.@]
690 * Make a permanent object temporary.
693 * Handle [I] handle to permanent object
696 * Success: STATUS_SUCCESS.
697 * Failure: An NTSTATUS error code.
699 NTSTATUS WINAPI
NtMakeTemporaryObject( HANDLE Handle
)
701 FIXME("(%p), stub.\n", Handle
);
702 return STATUS_SUCCESS
;