1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS system libraries 4 * FILE: dll/win32/kernel32/client/file/mntpoint.c 5 * PURPOSE: File volume mount point functions 6 * PROGRAMMER: Pierre Schweitzer (pierre@reactos.org) 7 */ 8 9 #include <k32.h> 10 #define NDEBUG 11 #include <debug.h> 12 13 /* 14 * @implemented 15 */ 16 static BOOL 17 GetVolumeNameForRoot(IN LPCWSTR lpszRootPath, 18 OUT LPWSTR lpszVolumeName, 19 IN DWORD cchBufferLength) 20 { 21 BOOL Ret; 22 NTSTATUS Status; 23 PWSTR FoundVolume; 24 DWORD BytesReturned; 25 UNICODE_STRING NtPathName; 26 IO_STATUS_BLOCK IoStatusBlock; 27 PMOUNTMGR_MOUNT_POINT MountPoint; 28 ULONG CurrentMntPt, FoundVolumeLen; 29 PMOUNTMGR_MOUNT_POINTS MountPoints; 30 OBJECT_ATTRIBUTES ObjectAttributes; 31 HANDLE VolumeHandle, MountMgrHandle; 32 struct 33 { 34 MOUNTDEV_NAME; 35 WCHAR Buffer[MAX_PATH]; 36 } MountDevName; 37 38 /* It makes no sense on a non-local drive */ 39 if (GetDriveTypeW(lpszRootPath) == DRIVE_REMOTE) 40 { 41 SetLastError(ERROR_PATH_NOT_FOUND); 42 return FALSE; 43 } 44 45 /* Get the NT path */ 46 if (!RtlDosPathNameToNtPathName_U(lpszRootPath, &NtPathName, NULL, NULL)) 47 { 48 SetLastError(ERROR_PATH_NOT_FOUND); 49 return FALSE; 50 } 51 52 /* If it's a root path - likely - drop backslash to open volume */ 53 if (NtPathName.Buffer[(NtPathName.Length / sizeof(WCHAR)) - 1] == L'\\') 54 { 55 NtPathName.Buffer[(NtPathName.Length / sizeof(WCHAR)) - 1] = UNICODE_NULL; 56 NtPathName.Length -= sizeof(WCHAR); 57 } 58 59 /* If that's a DOS volume, upper case the letter */ 60 if (NtPathName.Length >= 2 * sizeof(WCHAR)) 61 { 62 if (NtPathName.Buffer[(NtPathName.Length / sizeof(WCHAR)) - 1] == L':') 63 { 64 NtPathName.Buffer[(NtPathName.Length / sizeof(WCHAR)) - 2] = _toupper(NtPathName.Buffer[(NtPathName.Length / sizeof(WCHAR)) - 2]); 65 } 66 } 67 68 /* Attempt to open the volume */ 69 InitializeObjectAttributes(&ObjectAttributes, &NtPathName, 70 OBJ_CASE_INSENSITIVE, NULL, NULL); 71 Status = NtOpenFile(&VolumeHandle, SYNCHRONIZE | FILE_READ_ATTRIBUTES, 72 &ObjectAttributes, &IoStatusBlock, 73 FILE_SHARE_READ | FILE_SHARE_WRITE, 74 FILE_SYNCHRONOUS_IO_ALERT); 75 if (!NT_SUCCESS(Status)) 76 { 77 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 78 BaseSetLastNTError(Status); 79 return FALSE; 80 } 81 82 /* Query the device name - that's what we'll translate */ 83 if (!DeviceIoControl(VolumeHandle, IOCTL_MOUNTDEV_QUERY_DEVICE_NAME, NULL, 84 0, &MountDevName, sizeof(MountDevName), &BytesReturned, 85 NULL)) 86 { 87 NtClose(VolumeHandle); 88 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 89 return FALSE; 90 } 91 92 /* No longer need the volume */ 93 NtClose(VolumeHandle); 94 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 95 96 /* We'll keep the device name for later usage */ 97 NtPathName.Length = MountDevName.NameLength; 98 NtPathName.MaximumLength = MountDevName.NameLength + sizeof(UNICODE_NULL); 99 NtPathName.Buffer = RtlAllocateHeap(RtlGetProcessHeap(), 0, NtPathName.MaximumLength); 100 if (NtPathName.Buffer == NULL) 101 { 102 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 103 return FALSE; 104 } 105 106 RtlCopyMemory(NtPathName.Buffer, MountDevName.Name, NtPathName.Length); 107 NtPathName.Buffer[NtPathName.Length / sizeof(WCHAR)] = UNICODE_NULL; 108 109 /* Allocate the structure for querying the mount mgr */ 110 MountPoint = RtlAllocateHeap(RtlGetProcessHeap(), 0, 111 NtPathName.Length + sizeof(MOUNTMGR_MOUNT_POINT)); 112 if (MountPoint == NULL) 113 { 114 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 115 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 116 return FALSE; 117 } 118 119 /* 0 everything, we provide a device name */ 120 RtlZeroMemory(MountPoint, sizeof(MOUNTMGR_MOUNT_POINT)); 121 MountPoint->DeviceNameOffset = sizeof(MOUNTMGR_MOUNT_POINT); 122 MountPoint->DeviceNameLength = NtPathName.Length; 123 RtlCopyMemory((PVOID)((ULONG_PTR)MountPoint + sizeof(MOUNTMGR_MOUNT_POINT)), NtPathName.Buffer, NtPathName.Length); 124 125 /* Allocate a dummy output buffer to probe for size */ 126 MountPoints = RtlAllocateHeap(RtlGetProcessHeap(), 0, sizeof(MOUNTMGR_MOUNT_POINTS)); 127 if (MountPoints == NULL) 128 { 129 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 130 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoint); 131 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 132 return FALSE; 133 } 134 135 /* Open a handle to the mount manager */ 136 MountMgrHandle = CreateFileW(MOUNTMGR_DOS_DEVICE_NAME, 0, 137 FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, 138 OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 139 INVALID_HANDLE_VALUE); 140 if (MountMgrHandle == INVALID_HANDLE_VALUE) 141 { 142 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 143 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoints); 144 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoint); 145 return FALSE; 146 } 147 148 /* Query the names associated to our device name */ 149 Ret = DeviceIoControl(MountMgrHandle, IOCTL_MOUNTMGR_QUERY_POINTS, 150 MountPoint, NtPathName.Length + sizeof(MOUNTMGR_MOUNT_POINT), 151 MountPoints, sizeof(MOUNTMGR_MOUNT_POINTS), &BytesReturned, 152 NULL); 153 /* As long as the buffer is too small, keep looping */ 154 while (!Ret && GetLastError() == ERROR_MORE_DATA) 155 { 156 ULONG BufferSize; 157 158 /* Get the size we've to allocate */ 159 BufferSize = MountPoints->Size; 160 /* Reallocate the buffer with enough room */ 161 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoints); 162 MountPoints = RtlAllocateHeap(RtlGetProcessHeap(), 0, BufferSize); 163 if (MountPoints == NULL) 164 { 165 CloseHandle(MountMgrHandle); 166 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 167 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoint); 168 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 169 return FALSE; 170 } 171 172 /* Reissue the request, it should work now! */ 173 Ret = DeviceIoControl(MountMgrHandle, IOCTL_MOUNTMGR_QUERY_POINTS, 174 MountPoint, NtPathName.Length + sizeof(MOUNTMGR_MOUNT_POINT), 175 MountPoints, BufferSize, &BytesReturned, NULL); 176 } 177 178 /* We're done, no longer need the mount manager */ 179 CloseHandle(MountMgrHandle); 180 /* Nor our input buffer */ 181 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoint); 182 183 /* If the mount manager failed, just quit */ 184 if (!Ret) 185 { 186 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 187 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoints); 188 SetLastError(ERROR_INVALID_PARAMETER); 189 return FALSE; 190 } 191 192 CurrentMntPt = 0; 193 /* If there were no associated mount points, we'll return the device name */ 194 if (MountPoints->NumberOfMountPoints == 0) 195 { 196 FoundVolume = NtPathName.Buffer; 197 FoundVolumeLen = NtPathName.Length; 198 } 199 /* Otherwise, find one which is matching */ 200 else 201 { 202 for (; CurrentMntPt < MountPoints->NumberOfMountPoints; ++CurrentMntPt) 203 { 204 UNICODE_STRING SymbolicLink; 205 206 /* Make a string of it, to easy the checks */ 207 SymbolicLink.Length = MountPoints->MountPoints[CurrentMntPt].SymbolicLinkNameLength; 208 SymbolicLink.MaximumLength = SymbolicLink.Length; 209 SymbolicLink.Buffer = (PVOID)((ULONG_PTR)MountPoints + MountPoints->MountPoints[CurrentMntPt].SymbolicLinkNameOffset); 210 /* If that's a NT volume name (GUID form), keep it! */ 211 if (MOUNTMGR_IS_NT_VOLUME_NAME(&SymbolicLink)) 212 { 213 FoundVolume = SymbolicLink.Buffer; 214 FoundVolumeLen = SymbolicLink.Length; 215 216 break; 217 } 218 } 219 } 220 221 /* We couldn't find anything matching, return an error */ 222 if (CurrentMntPt == MountPoints->NumberOfMountPoints) 223 { 224 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 225 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoints); 226 SetLastError(ERROR_INVALID_PARAMETER); 227 return FALSE; 228 } 229 230 /* We found a matching volume, have we enough memory to return it? */ 231 if (cchBufferLength * sizeof(WCHAR) < FoundVolumeLen + 2 * sizeof(WCHAR)) 232 { 233 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 234 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoints); 235 SetLastError(ERROR_FILENAME_EXCED_RANGE); 236 return FALSE; 237 } 238 239 /* Copy it back! */ 240 RtlCopyMemory(lpszVolumeName, FoundVolume, FoundVolumeLen); 241 /* Make it compliant */ 242 lpszVolumeName[1] = L'\\'; 243 /* And transform it as root path */ 244 lpszVolumeName[FoundVolumeLen / sizeof(WCHAR)] = L'\\'; 245 lpszVolumeName[FoundVolumeLen / sizeof(WCHAR) + 1] = UNICODE_NULL; 246 247 /* We're done! */ 248 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 249 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoints); 250 return TRUE; 251 } 252 253 /* 254 * @implemented 255 */ 256 BOOL 257 BasepGetVolumeNameFromReparsePoint(IN LPCWSTR lpszMountPoint, 258 OUT LPWSTR lpszVolumeName, 259 IN DWORD cchBufferLength, 260 OUT LPBOOL IsAMountPoint) 261 { 262 WCHAR Old; 263 DWORD BytesReturned; 264 HANDLE ReparseHandle; 265 UNICODE_STRING SubstituteName; 266 PREPARSE_DATA_BUFFER ReparseBuffer; 267 268 /* Try to open the reparse point */ 269 ReparseHandle = CreateFileW(lpszMountPoint, 0, 270 FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, 271 OPEN_EXISTING, 272 FILE_FLAG_OPEN_REPARSE_POINT | FILE_FLAG_BACKUP_SEMANTICS | FILE_ATTRIBUTE_NORMAL, 273 INVALID_HANDLE_VALUE); 274 /* It failed! */ 275 if (ReparseHandle == INVALID_HANDLE_VALUE) 276 { 277 /* Report it's not a mount point (it's not a reparse point) */ 278 if (IsAMountPoint != NULL) 279 { 280 *IsAMountPoint = FALSE; 281 } 282 283 /* And zero output */ 284 if (lpszVolumeName != NULL && cchBufferLength >= 1) 285 { 286 lpszVolumeName[0] = UNICODE_NULL; 287 } 288 289 return FALSE; 290 } 291 292 /* This is a mount point! */ 293 if (IsAMountPoint != NULL) 294 { 295 *IsAMountPoint = TRUE; 296 } 297 298 /* Prepare a buffer big enough to read its data */ 299 ReparseBuffer = RtlAllocateHeap(RtlGetProcessHeap(), 0, MAXIMUM_REPARSE_DATA_BUFFER_SIZE); 300 if (ReparseBuffer == NULL) 301 { 302 CloseHandle(ReparseHandle); 303 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 304 305 /* Zero output */ 306 if (lpszVolumeName != NULL && cchBufferLength >= 1) 307 { 308 lpszVolumeName[0] = UNICODE_NULL; 309 } 310 311 return FALSE; 312 } 313 314 /* Dump the reparse point data */ 315 if (!DeviceIoControl(ReparseHandle, FSCTL_GET_REPARSE_POINT, NULL, 0, 316 ReparseBuffer, MAXIMUM_REPARSE_DATA_BUFFER_SIZE, &BytesReturned, 317 NULL)) 318 { 319 RtlFreeHeap(RtlGetProcessHeap(), 0, ReparseBuffer); 320 CloseHandle(ReparseHandle); 321 322 /* Zero output */ 323 if (lpszVolumeName != NULL && cchBufferLength >= 1) 324 { 325 lpszVolumeName[0] = UNICODE_NULL; 326 } 327 328 return FALSE; 329 } 330 331 /* We no longer need the reparse point */ 332 CloseHandle(ReparseHandle); 333 334 /* We only handle mount points */ 335 if (ReparseBuffer->ReparseTag != IO_REPARSE_TAG_MOUNT_POINT) 336 { 337 RtlFreeHeap(RtlGetProcessHeap(), 0, ReparseBuffer); 338 339 /* Zero output */ 340 if (lpszVolumeName != NULL && cchBufferLength >= 1) 341 { 342 lpszVolumeName[0] = UNICODE_NULL; 343 } 344 345 return FALSE; 346 } 347 348 /* Do we have enough room for copying substitue name? */ 349 if ((ReparseBuffer->MountPointReparseBuffer.SubstituteNameLength + sizeof(UNICODE_NULL)) > cchBufferLength * sizeof(WCHAR)) 350 { 351 RtlFreeHeap(RtlGetProcessHeap(), 0, ReparseBuffer); 352 SetLastError(ERROR_FILENAME_EXCED_RANGE); 353 354 /* Zero output */ 355 if (lpszVolumeName != NULL && cchBufferLength >= 1) 356 { 357 lpszVolumeName[0] = UNICODE_NULL; 358 } 359 360 return FALSE; 361 } 362 363 /* Copy the link target */ 364 RtlCopyMemory(lpszVolumeName, 365 &ReparseBuffer->MountPointReparseBuffer.PathBuffer[ReparseBuffer->MountPointReparseBuffer.SubstituteNameOffset / sizeof(WCHAR)], 366 ReparseBuffer->MountPointReparseBuffer.SubstituteNameLength); 367 /* Make it DOS valid */ 368 Old = lpszVolumeName[1]; 369 /* We want a root path */ 370 lpszVolumeName[1] = L'\\'; 371 /* And null terminate obviously */ 372 lpszVolumeName[ReparseBuffer->MountPointReparseBuffer.SubstituteNameLength / sizeof(WCHAR)] = UNICODE_NULL; 373 374 /* Make it a string to easily check it */ 375 SubstituteName.Length = ReparseBuffer->MountPointReparseBuffer.SubstituteNameLength; 376 SubstituteName.MaximumLength = SubstituteName.Length; 377 SubstituteName.Buffer = lpszVolumeName; 378 379 /* No longer need the data? */ 380 RtlFreeHeap(RtlGetProcessHeap(), 0, ReparseBuffer); 381 382 /* Is that a dos volume name with backslash? */ 383 if (MOUNTMGR_IS_DOS_VOLUME_NAME_WB(&SubstituteName)) 384 { 385 return TRUE; 386 } 387 388 /* No, so restore previous name and return to the caller */ 389 lpszVolumeName[1] = Old; 390 SetLastError(ERROR_INVALID_PARAMETER); 391 return FALSE; 392 } 393 394 /* 395 * @implemented 396 */ 397 BOOL 398 BasepGetVolumeNameForVolumeMountPoint(IN LPCWSTR lpszMountPoint, 399 OUT LPWSTR lpszVolumeName, 400 IN DWORD cchBufferLength, 401 OUT LPBOOL IsAMountPoint) 402 { 403 BOOL Ret; 404 UNICODE_STRING MountPoint; 405 406 /* Assume it's a mount point (likely for non reparse points) */ 407 if (IsAMountPoint != NULL) 408 { 409 *IsAMountPoint = 1; 410 } 411 412 /* Make a string with the mount point name */ 413 RtlInitUnicodeString(&MountPoint, lpszMountPoint); 414 /* Not a root path? */ 415 if (MountPoint.Buffer[(MountPoint.Length / sizeof(WCHAR)) - 1] != L'\\') 416 { 417 BaseSetLastNTError(STATUS_OBJECT_NAME_INVALID); 418 /* Zero output */ 419 if (lpszVolumeName != NULL && cchBufferLength >= 1) 420 { 421 lpszVolumeName[0] = UNICODE_NULL; 422 } 423 424 return FALSE; 425 } 426 427 /* Does it look like <letter>:\? */ 428 if (MountPoint.Length == 3 * sizeof(WCHAR)) 429 { 430 /* Try to get volume name for root path */ 431 Ret = GetVolumeNameForRoot(lpszMountPoint, lpszVolumeName, cchBufferLength); 432 /* It failed? */ 433 if (!Ret) 434 { 435 /* If wasn't a drive letter, so maybe a reparse point? */ 436 if (MountPoint.Buffer[1] != ':') 437 { 438 Ret = BasepGetVolumeNameFromReparsePoint(lpszMountPoint, lpszVolumeName, cchBufferLength, IsAMountPoint); 439 } 440 /* It was, so zero output */ 441 else if (lpszVolumeName != NULL && cchBufferLength >= 1) 442 { 443 lpszVolumeName[0] = UNICODE_NULL; 444 } 445 } 446 } 447 else 448 { 449 /* Try to get volume name for root path */ 450 Ret = GetVolumeNameForRoot(lpszMountPoint, lpszVolumeName, cchBufferLength); 451 /* It failed? */ 452 if (!Ret) 453 { 454 /* It was a DOS volume as UNC name, so fail and zero output */ 455 if (MountPoint.Length == 14 && MountPoint.Buffer[0] == '\\' && MountPoint.Buffer[1] == '\\' && 456 (MountPoint.Buffer[2] == '.' || MountPoint.Buffer[2] == '?') && MountPoint.Buffer[3] == L'\\' && 457 MountPoint.Buffer[5] == ':') 458 { 459 if (lpszVolumeName != NULL && cchBufferLength >= 1) 460 { 461 lpszVolumeName[0] = UNICODE_NULL; 462 } 463 } 464 /* Maybe it's a reparse point? */ 465 else 466 { 467 Ret = BasepGetVolumeNameFromReparsePoint(lpszMountPoint, lpszVolumeName, cchBufferLength, IsAMountPoint); 468 } 469 } 470 } 471 472 return Ret; 473 } 474 475 /** 476 * @name GetVolumeNameForVolumeMountPointW 477 * @implemented 478 * 479 * Return an unique volume name for a drive root or mount point. 480 * 481 * @param VolumeMountPoint 482 * Pointer to string that contains either root drive name or 483 * mount point name. 484 * @param VolumeName 485 * Pointer to buffer that is filled with resulting unique 486 * volume name on success. 487 * @param VolumeNameLength 488 * Size of VolumeName buffer in TCHARs. 489 * 490 * @return 491 * TRUE when the function succeeds and the VolumeName buffer is filled, 492 * FALSE otherwise. 493 */ 494 BOOL 495 WINAPI 496 GetVolumeNameForVolumeMountPointW(IN LPCWSTR VolumeMountPoint, 497 OUT LPWSTR VolumeName, 498 IN DWORD VolumeNameLength) 499 { 500 BOOL Ret; 501 502 /* Just query our internal function */ 503 Ret = BasepGetVolumeNameForVolumeMountPoint(VolumeMountPoint, VolumeName, 504 VolumeNameLength, NULL); 505 if (!Ret && VolumeName != NULL && VolumeNameLength >= 1) 506 { 507 VolumeName[0] = UNICODE_NULL; 508 } 509 510 return Ret; 511 } 512 513 /* 514 * @implemented 515 */ 516 BOOL 517 WINAPI 518 GetVolumeNameForVolumeMountPointA(IN LPCSTR lpszVolumeMountPoint, 519 IN LPSTR lpszVolumeName, 520 IN DWORD cchBufferLength) 521 { 522 BOOL Ret; 523 ANSI_STRING VolumeName; 524 UNICODE_STRING VolumeNameU; 525 PUNICODE_STRING VolumeMountPointU; 526 527 /* Convert mount point to unicode */ 528 VolumeMountPointU = Basep8BitStringToStaticUnicodeString(lpszVolumeMountPoint); 529 if (VolumeMountPointU == NULL) 530 { 531 return FALSE; 532 } 533 534 /* Initialize the strings we'll use for convention */ 535 VolumeName.Buffer = lpszVolumeName; 536 VolumeName.Length = 0; 537 VolumeName.MaximumLength = cchBufferLength - 1; 538 539 VolumeNameU.Length = 0; 540 VolumeNameU.MaximumLength = (cchBufferLength - 1) * sizeof(WCHAR) + sizeof(UNICODE_NULL); 541 /* Allocate a buffer big enough to contain the returned name */ 542 VolumeNameU.Buffer = RtlAllocateHeap(RtlGetProcessHeap(), 0, VolumeNameU.MaximumLength); 543 if (VolumeNameU.Buffer == NULL) 544 { 545 SetLastError(ERROR_NOT_ENOUGH_MEMORY); 546 return FALSE; 547 } 548 549 /* Query -W */ 550 Ret = GetVolumeNameForVolumeMountPointW(VolumeMountPointU->Buffer, VolumeNameU.Buffer, cchBufferLength); 551 /* If it succeed, perform -A conversion */ 552 if (Ret) 553 { 554 NTSTATUS Status; 555 556 /* Reinit our string for length */ 557 RtlInitUnicodeString(&VolumeNameU, VolumeNameU.Buffer); 558 /* Convert to ANSI */ 559 Status = RtlUnicodeStringToAnsiString(&VolumeName, &VolumeNameU, FALSE); 560 /* If conversion failed, force failure, otherwise, just null terminate */ 561 if (!NT_SUCCESS(Status)) 562 { 563 Ret = FALSE; 564 BaseSetLastNTError(Status); 565 } 566 else 567 { 568 VolumeName.Buffer[VolumeName.Length] = ANSI_NULL; 569 } 570 } 571 572 /* Internal buffer no longer needed */ 573 RtlFreeHeap(RtlGetProcessHeap(), 0, VolumeNameU.Buffer); 574 575 return Ret; 576 } 577 578 /* 579 * @unimplemented 580 */ 581 BOOL 582 WINAPI 583 SetVolumeMountPointW(IN LPCWSTR lpszVolumeMountPoint, 584 IN LPCWSTR lpszVolumeName) 585 { 586 STUB; 587 return 0; 588 } 589 590 /* 591 * @unimplemented 592 */ 593 BOOL 594 WINAPI 595 SetVolumeMountPointA(IN LPCSTR lpszVolumeMountPoint, 596 IN LPCSTR lpszVolumeName) 597 { 598 STUB; 599 return 0; 600 } 601 602 /* 603 * @unimplemented 604 */ 605 BOOL 606 WINAPI 607 DeleteVolumeMountPointA(IN LPCSTR lpszVolumeMountPoint) 608 { 609 STUB; 610 return 0; 611 } 612 613 /* 614 * @unimplemented 615 */ 616 BOOL 617 WINAPI 618 DeleteVolumeMountPointW(IN LPCWSTR lpszVolumeMountPoint) 619 { 620 STUB; 621 return 0; 622 } 623 624 /* 625 * @unimplemented 626 */ 627 HANDLE 628 WINAPI 629 FindFirstVolumeMountPointW(IN LPCWSTR lpszRootPathName, 630 IN LPWSTR lpszVolumeMountPoint, 631 IN DWORD cchBufferLength) 632 { 633 STUB; 634 return 0; 635 } 636 637 /* 638 * @unimplemented 639 */ 640 HANDLE 641 WINAPI 642 FindFirstVolumeMountPointA(IN LPCSTR lpszRootPathName, 643 IN LPSTR lpszVolumeMountPoint, 644 IN DWORD cchBufferLength) 645 { 646 STUB; 647 return 0; 648 } 649 650 /* 651 * @unimplemented 652 */ 653 BOOL 654 WINAPI 655 FindNextVolumeMountPointA(IN HANDLE hFindVolumeMountPoint, 656 IN LPSTR lpszVolumeMountPoint, 657 DWORD cchBufferLength) 658 { 659 STUB; 660 return 0; 661 } 662 663 /* 664 * @unimplemented 665 */ 666 BOOL 667 WINAPI 668 FindNextVolumeMountPointW(IN HANDLE hFindVolumeMountPoint, 669 IN LPWSTR lpszVolumeMountPoint, 670 DWORD cchBufferLength) 671 { 672 STUB; 673 return 0; 674 } 675 676 /* 677 * @unimplemented 678 */ 679 BOOL 680 WINAPI 681 FindVolumeMountPointClose(IN HANDLE hFindVolumeMountPoint) 682 { 683 STUB; 684 return 0; 685 } 686 687 /* EOF */ 688