1 // ClientConnection.cpp 2 3 #include "ClientConnection.h" 4 5 #include <new> 6 #include <typeinfo> 7 8 #include <dirent.h> 9 #include <fcntl.h> 10 #include <errno.h> 11 #include <stdlib.h> 12 #include <unistd.h> 13 #include <utime.h> 14 15 #include <AutoDeleter.h> 16 #include <AutoLocker.h> 17 #include <Entry.h> 18 #include <fs_query.h> 19 #include <GraphicsDefs.h> 20 #include <HashMap.h> 21 #include <NodeMonitor.h> 22 #include <Path.h> 23 #include <Rect.h> 24 #include <Mime.h> 25 26 #include <fsproto.h> 27 28 #include "Compatibility.h" 29 #include "Connection.h" 30 #include "DebugSupport.h" 31 #include "Directory.h" 32 #include "Entry.h" 33 #include "FDManager.h" 34 #include "NodeHandle.h" 35 #include "NodeHandleMap.h" 36 #include "NodeMonitoringEvent.h" 37 #include "Path.h" 38 #include "RequestBufferReplacer.h" 39 #include "RequestChannel.h" 40 #include "RequestConnection.h" 41 #include "RequestDumper.h" 42 #include "RequestFlattener.h" 43 #include "Requests.h" 44 #include "SecurityContext.h" 45 #include "ServerNodeID.h" 46 #include "UserSecurityContext.h" 47 #include "Utils.h" 48 #include "Volume.h" 49 #include "VolumeManager.h" 50 51 static const int32 kMaxSaneReadLinkSize = 10240; // 10 KB 52 static const int32 kMaxReadBufferSize = 10240; // 10 KB 53 static const int32 kMaxReadDirBufferSize = 10240; 54 55 // Locking: 56 // 57 // fLock: Guards fReferenceCount and fClosed. 58 // fSecurityContextLock: Guards fSecurityContext. 59 // fVolumes: Guards the map itself. 60 61 62 // #pragma mark - 63 // #pragma mark ----- ClientConnection ----- 64 65 // ConnectionReference 66 class ClientConnection::ConnectionReference { 67 public: 68 ConnectionReference(ClientConnection* connection) 69 : fConnection(connection) 70 { 71 if (!fConnection || !fConnection->GetReference()) 72 fConnection = NULL; 73 } 74 75 ~ConnectionReference() 76 { 77 if (fConnection) 78 fConnection->PutReference(); 79 } 80 81 bool IsValid() const 82 { 83 return fConnection; 84 } 85 86 private: 87 ClientConnection* fConnection; 88 }; 89 90 // VolumeMap 91 struct ClientConnection::VolumeMap 92 : public SynchronizedHashMap<HashKey32<int32>, ClientVolume*> { 93 }; 94 95 // ClientVolumePutter 96 class ClientConnection::ClientVolumePutter { 97 public: 98 ClientVolumePutter(ClientConnection* connection, ClientVolume* volume) 99 : fConnection(connection), 100 fVolume(volume) 101 { 102 } 103 104 ~ClientVolumePutter() 105 { 106 if (fConnection && fVolume) 107 fConnection->_PutVolume(fVolume); 108 } 109 110 void Detach() 111 { 112 fConnection = NULL; 113 fVolume = NULL; 114 } 115 116 private: 117 ClientConnection* fConnection; 118 ClientVolume* fVolume; 119 }; 120 121 // VolumeNodeMonitoringEvent 122 struct ClientConnection::VolumeNodeMonitoringEvent { 123 VolumeNodeMonitoringEvent(int32 volumeID, NodeMonitoringEvent* event) 124 : volumeID(volumeID), 125 event(event) 126 { 127 if (event) 128 event->AcquireReference(); 129 } 130 131 ~VolumeNodeMonitoringEvent() 132 { 133 if (event) 134 event->ReleaseReference(); 135 } 136 137 int32 volumeID; 138 NodeMonitoringEvent* event; 139 }; 140 141 // NodeMonitoringEventQueue 142 struct ClientConnection::NodeMonitoringEventQueue 143 : BlockingQueue<NodeMonitoringRequest> { 144 NodeMonitoringEventQueue() 145 : BlockingQueue<NodeMonitoringRequest>("client NM requests") 146 { 147 } 148 }; 149 150 // QueryHandleUnlocker 151 struct ClientConnection::QueryHandleUnlocker { 152 QueryHandleUnlocker(ClientConnection* connection, NodeHandle* nodeHandle) 153 : fConnection(connection), 154 fHandle(nodeHandle) 155 { 156 } 157 158 ~QueryHandleUnlocker() 159 { 160 if (fConnection && fHandle) { 161 fConnection->_UnlockQueryHandle(fHandle); 162 fConnection = NULL; 163 fHandle = NULL; 164 } 165 } 166 167 private: 168 ClientConnection* fConnection; 169 NodeHandle* fHandle; 170 }; 171 172 // ClientVolumeFilter 173 struct ClientConnection::ClientVolumeFilter { 174 virtual ~ClientVolumeFilter() {} 175 176 virtual bool FilterVolume(ClientConnection* connection, 177 ClientVolume* volume) = 0; 178 }; 179 180 // HasQueryPermissionClientVolumeFilter 181 struct ClientConnection::HasQueryPermissionClientVolumeFilter 182 : ClientConnection::ClientVolumeFilter { 183 virtual bool FilterVolume(ClientConnection* connection, 184 ClientVolume* volume) 185 { 186 return volume->GetSharePermissions().ImpliesQuerySharePermission(); 187 } 188 }; 189 190 191 // #pragma mark - 192 193 // constructor 194 ClientConnection::ClientConnection(Connection* connection, 195 SecurityContext* securityContext, User* user, 196 ClientConnectionListener* listener) 197 : RequestHandler(), 198 ClientVolume::NodeMonitoringProcessor(), 199 fConnection(NULL), 200 fSecurityContext(securityContext), 201 fSecurityContextLock("security context lock"), 202 fUser(user), 203 fVolumes(NULL), 204 fQueryHandles(NULL), 205 fListener(listener), 206 fNodeMonitoringEvents(NULL), 207 fNodeMonitoringProcessor(-1), 208 fLock("client connection locker"), 209 fReferenceCount(0), 210 fInitialized(0), 211 fClosed(false), 212 fError(false), 213 fInverseClientEndianess(false) 214 { 215 fConnection = new(std::nothrow) RequestConnection(connection, this); 216 if (!fConnection) 217 delete connection; 218 } 219 220 // destructor 221 ClientConnection::~ClientConnection() 222 { 223 _Close(); 224 delete fConnection; 225 226 // delete all volumes 227 for (VolumeMap::Iterator it = fVolumes->GetIterator(); it.HasNext();) 228 delete it.Next().value; 229 delete fVolumes; 230 231 delete fQueryHandles; 232 delete fNodeMonitoringEvents; 233 } 234 235 // Init 236 status_t 237 ClientConnection::Init() 238 { 239 // create a client volume map 240 fVolumes = new(std::nothrow) VolumeMap; 241 if (!fVolumes) 242 return B_NO_MEMORY; 243 status_t error = fVolumes->InitCheck(); 244 if (error != B_OK) 245 return error; 246 247 // create the query handle map 248 fQueryHandles = new(std::nothrow) NodeHandleMap("query handles"); 249 if (!fQueryHandles) 250 return B_NO_MEMORY; 251 error = fQueryHandles->Init(); 252 if (error != B_OK) 253 return error; 254 255 // create the node monitoring event queue 256 fNodeMonitoringEvents = new(std::nothrow) NodeMonitoringEventQueue; 257 if (!fNodeMonitoringEvents) 258 return B_NO_MEMORY; 259 error = fNodeMonitoringEvents->InitCheck(); 260 if (error != B_OK) 261 return error; 262 263 // initialize the connection 264 error = fConnection->Init(); 265 if (error != B_OK) 266 return error; 267 268 // start the node monitoring processor 269 fNodeMonitoringProcessor = spawn_thread(_NodeMonitoringProcessorEntry, 270 "client connection NM processor", B_NORMAL_PRIORITY, this); 271 if (fNodeMonitoringProcessor < 0) { 272 _Close(); 273 return fNodeMonitoringProcessor; 274 } 275 resume_thread(fNodeMonitoringProcessor); 276 return B_OK; 277 } 278 279 // Close 280 /*! 281 Called by the NetFSServer. Not for internal use. Waits for the connection 282 to be closed (at least for the node monitoring thread to be gone). 283 */ 284 void 285 ClientConnection::Close() 286 { 287 { 288 ConnectionReference connectionReference(this); 289 if (connectionReference.IsValid()) 290 _MarkClosed(false); 291 fListener = NULL; 292 } 293 294 // Wait at least for the node monitoring processor; this is not perfect, 295 // but not too bad either. 296 if (fNodeMonitoringProcessor >= 0 297 && find_thread(NULL) != fNodeMonitoringProcessor) { 298 int32 result; 299 wait_for_thread(fNodeMonitoringProcessor, &result); 300 } 301 } 302 303 // GetReference 304 bool 305 ClientConnection::GetReference() 306 { 307 AutoLocker<Locker> _(fLock); 308 if (fClosed || !atomic_or(&fInitialized, 0)) 309 return false; 310 fReferenceCount++; 311 return true; 312 } 313 314 // PutReference 315 void 316 ClientConnection::PutReference() 317 { 318 bool close = false; 319 { 320 AutoLocker<Locker> _(fLock); 321 --fReferenceCount; 322 if (fClosed) 323 close = (fReferenceCount == 0); 324 } 325 if (close) 326 _Close(); 327 } 328 329 // UserRemoved 330 void 331 ClientConnection::UserRemoved(User* user) 332 { 333 // get all volumes 334 ClientVolume** volumes = NULL; 335 int32 volumeCount = 0; 336 AutoLocker<VolumeMap> volumesLocker(fVolumes); 337 volumes = new(std::nothrow) ClientVolume*[fVolumes->Size()]; 338 if (!volumes) { 339 ERROR(("ClientConnection::UserRemoved(): ERROR: Failed to " 340 "allocate memory for volume array.\n")); 341 volumesLocker.Unlock(); 342 _UnmountAllVolumes(); 343 return; 344 } 345 for (VolumeMap::Iterator it = fVolumes->GetIterator(); it.HasNext();) { 346 if (ClientVolume* volume = _GetVolume(it.Next().value->GetID())) 347 volumes[volumeCount++] = volume; 348 } 349 volumesLocker.Unlock(); 350 351 // unmount the concerned volumes 352 for (int32 i = 0; i < volumeCount; i++) { 353 ClientVolume* volume = volumes[i]; 354 355 fSecurityContextLock.Lock(); 356 bool unmount = (volume->GetSecurityContext()->GetUser() == user); 357 fSecurityContextLock.Unlock(); 358 359 if (unmount) 360 _UnmountVolume(volume); 361 } 362 363 // put the volumes 364 for (int32 i = 0; i < volumeCount; i++) 365 _PutVolume(volumes[i]); 366 delete[] volumes; 367 } 368 369 // ShareRemoved 370 void 371 ClientConnection::ShareRemoved(Share* share) 372 { 373 // get all volumes 374 ClientVolume** volumes = NULL; 375 int32 volumeCount = 0; 376 AutoLocker<VolumeMap> volumesLocker(fVolumes); 377 volumes = new(std::nothrow) ClientVolume*[fVolumes->Size()]; 378 if (!volumes) { 379 ERROR(("ClientConnection::ShareRemoved(): ERROR: Failed to " 380 "allocate memory for volume array.\n")); 381 volumesLocker.Unlock(); 382 _UnmountAllVolumes(); 383 return; 384 } 385 for (VolumeMap::Iterator it = fVolumes->GetIterator(); it.HasNext();) { 386 if (ClientVolume* volume = _GetVolume(it.Next().value->GetID())) 387 volumes[volumeCount++] = volume; 388 } 389 volumesLocker.Unlock(); 390 391 // unmount the concerned volumes 392 for (int32 i = 0; i < volumeCount; i++) { 393 ClientVolume* volume = volumes[i]; 394 395 fSecurityContextLock.Lock(); 396 bool unmount = (volume->GetShare() == share); 397 fSecurityContextLock.Unlock(); 398 399 if (unmount) 400 _UnmountVolume(volume); 401 } 402 403 // put the volumes 404 for (int32 i = 0; i < volumeCount; i++) 405 _PutVolume(volumes[i]); 406 delete[] volumes; 407 } 408 409 // UserPermissionsChanged 410 void 411 ClientConnection::UserPermissionsChanged(Share* share, User* user, 412 Permissions permissions) 413 { 414 bool unmountAll = (!permissions.ImpliesMountSharePermission()); 415 416 // get all volumes 417 ClientVolume** volumes = NULL; 418 int32 volumeCount = 0; 419 AutoLocker<VolumeMap> volumesLocker(fVolumes); 420 volumes = new(std::nothrow) ClientVolume*[fVolumes->Size()]; 421 if (!volumes) { 422 ERROR(("ClientConnection::ShareRemoved(): ERROR: Failed to " 423 "allocate memory for volume array.\n")); 424 volumesLocker.Unlock(); 425 _UnmountAllVolumes(); 426 return; 427 } 428 for (VolumeMap::Iterator it = fVolumes->GetIterator(); it.HasNext();) { 429 if (ClientVolume* volume = _GetVolume(it.Next().value->GetID())) 430 volumes[volumeCount++] = volume; 431 } 432 volumesLocker.Unlock(); 433 434 // update the concerned volumes 435 for (int32 i = 0; i < volumeCount; i++) { 436 ClientVolume* volume = volumes[i]; 437 438 fSecurityContextLock.Lock(); 439 bool concerned = (volume->GetShare() == share 440 && volume->GetSecurityContext()->GetUser() == user); 441 fSecurityContextLock.Unlock(); 442 443 if (concerned) { 444 // create a new user security context for the volume 445 status_t error = B_OK; 446 447 if (unmountAll) { 448 _UnmountVolume(volume); 449 } else { 450 // create a new user security context 451 AutoLocker<Locker> securityContextLocker(fSecurityContextLock); 452 UserSecurityContext* userSecurityContext 453 = new(std::nothrow) UserSecurityContext; 454 455 // init it 456 if (userSecurityContext) { 457 error = fSecurityContext->GetUserSecurityContext(user, 458 userSecurityContext); 459 } else 460 error = B_NO_MEMORY; 461 if (error != B_OK) { 462 delete userSecurityContext; 463 securityContextLocker.Unlock(); 464 _UnmountVolume(volume); 465 continue; 466 } 467 468 // set the volume's new user security context 469 securityContextLocker.Unlock(); 470 volume->SetSecurityContext(userSecurityContext); 471 } 472 } 473 } 474 475 // put the volumes 476 for (int32 i = 0; i < volumeCount; i++) 477 _PutVolume(volumes[i]); 478 delete[] volumes; 479 } 480 481 482 // #pragma mark - 483 484 // VisitConnectionBrokenRequest 485 status_t 486 ClientConnection::VisitConnectionBrokenRequest(ConnectionBrokenRequest* request) 487 { 488 ConnectionReference connectionReference(this); 489 if (!connectionReference.IsValid()) 490 return B_OK; 491 492 _MarkClosed(true); 493 return B_OK; 494 } 495 496 // VisitInitConnectionRequest 497 status_t 498 ClientConnection::VisitInitConnectionRequest(InitConnectionRequest* request) 499 { 500 bool alreadyInitialized = atomic_or(&fInitialized, ~0); 501 502 ConnectionReference connectionReference(this); 503 if (!connectionReference.IsValid()) 504 return B_OK; 505 506 if (!alreadyInitialized) 507 fInverseClientEndianess = (request->bigEndian != B_HOST_IS_BENDIAN); 508 509 // prepare the reply 510 InitConnectionReply reply; 511 512 // send the reply 513 reply.error = (alreadyInitialized ? B_BAD_VALUE : B_OK); 514 status_t error = GetChannel()->SendRequest(&reply); 515 516 // on error just close 517 if (error != B_OK) 518 _MarkClosed(true); 519 return B_OK; 520 } 521 522 // VisitMountRequest 523 status_t 524 ClientConnection::VisitMountRequest(MountRequest* request) 525 { 526 ConnectionReference connectionReference(this); 527 if (!connectionReference.IsValid()) 528 return B_OK; 529 530 status_t result = B_OK; 531 const char* shareName = request->share.GetString(); 532 if (!shareName) 533 SET_ERROR(result, B_BAD_DATA); 534 535 // create a volume 536 ClientVolume* volume = NULL; 537 if (result == B_OK) 538 result = _CreateVolume(&volume); 539 ClientVolumePutter volumePutter(this, volume); 540 541 // if we haven't been supplied with a user yet, use the info from the 542 // mount request for authentication 543 VolumeManagerLocker managerLocker; 544 AutoLocker<Locker> securityContextLocker(fSecurityContextLock); 545 const char* userName = request->user.GetString(); 546 User* user = fUser; 547 BReference<User> userReference(user); 548 bool noPermission = false; 549 if (result == B_OK && !user) { 550 if (userName) { 551 SET_ERROR(result, fSecurityContext->AuthenticateUser(userName, 552 request->password.GetString(), &user)); 553 if (result == B_OK) 554 userReference.SetTo(user, true); 555 } else 556 result = B_PERMISSION_DENIED; 557 if (result == B_PERMISSION_DENIED) 558 noPermission = true; 559 } 560 561 // create a user security context 562 UserSecurityContext* securityContext = NULL; 563 if (result == B_OK) { 564 securityContext = new(std::nothrow) UserSecurityContext; 565 if (securityContext) { 566 SET_ERROR(result, fSecurityContext->GetUserSecurityContext(user, 567 securityContext)); 568 } else 569 SET_ERROR(result, B_NO_MEMORY); 570 } 571 ObjectDeleter<UserSecurityContext> securityContextDeleter(securityContext); 572 573 // get the share 574 Share* share = NULL; 575 Permissions sharePermissions; 576 node_ref mountPoint; 577 if (result == B_OK) { 578 AutoLocker<SecurityContext> _(fSecurityContext); 579 share = fSecurityContext->FindShare(shareName); 580 if (share) { 581 mountPoint = share->GetNodeRef(); 582 sharePermissions = securityContext->GetNodePermissions( 583 mountPoint); 584 if (!sharePermissions.ImpliesMountSharePermission()) { 585 SET_ERROR(result, B_PERMISSION_DENIED); 586 noPermission = true; 587 } 588 } else 589 SET_ERROR(result, B_ENTRY_NOT_FOUND); 590 } 591 BReference<Share> shareReference(share, true); 592 593 // mount the volume 594 MountReply reply; 595 if (result == B_OK) { 596 SET_ERROR(result, volume->Mount(securityContext, share)); 597 securityContextDeleter.Detach(); 598 } 599 if (result == B_OK) { 600 _GetNodeInfo(volume->GetRootDirectory(), &reply.nodeInfo); 601 reply.sharePermissions = sharePermissions.GetPermissions(); 602 reply.volumeID = volume->GetID(); 603 } 604 605 // make sure, the volume is removed on error 606 if (result != B_OK && volume) { 607 AutoLocker<VolumeMap> volumeMapLocker(fVolumes); 608 volume->MarkRemoved(); 609 } 610 611 securityContextLocker.Unlock(); 612 managerLocker.Unlock(); 613 614 // send the reply 615 reply.error = result; 616 reply.noPermission = noPermission; 617 return GetChannel()->SendRequest(&reply); 618 } 619 620 // VisitUnmountRequest 621 status_t 622 ClientConnection::VisitUnmountRequest(UnmountRequest* request) 623 { 624 ConnectionReference connectionReference(this); 625 if (!connectionReference.IsValid()) 626 return B_OK; 627 628 if (ClientVolume* volume = _GetVolume(request->volumeID)) { 629 _UnmountVolume(volume); 630 _PutVolume(volume); 631 } 632 633 return B_OK; 634 } 635 636 // VisitReadVNodeRequest 637 status_t 638 ClientConnection::VisitReadVNodeRequest(ReadVNodeRequest* request) 639 { 640 ConnectionReference connectionReference(this); 641 if (!connectionReference.IsValid()) 642 return B_OK; 643 644 // get the volume 645 status_t result = B_OK; 646 ClientVolume* volume = _GetVolume(request->volumeID); 647 if (!volume) 648 result = B_BAD_VALUE; 649 ClientVolumePutter volumePutter(this, volume); 650 651 VolumeManagerLocker managerLocker; 652 653 // get the node 654 Node* node = NULL; 655 if (result == B_OK) { 656 node = volume->GetNode(request->nodeID); 657 if (!node) 658 result = B_ENTRY_NOT_FOUND; 659 } 660 661 // prepare the reply 662 ReadVNodeReply reply; 663 if (result == B_OK) 664 _GetNodeInfo(node, &reply.nodeInfo); 665 666 managerLocker.Unlock(); 667 668 // send the reply 669 reply.error = result; 670 return GetChannel()->SendRequest(&reply); 671 } 672 673 // VisitWriteStatRequest 674 status_t 675 ClientConnection::VisitWriteStatRequest(WriteStatRequest* request) 676 { 677 ConnectionReference connectionReference(this); 678 if (!connectionReference.IsValid()) 679 return B_OK; 680 681 // get the volume 682 status_t result = B_OK; 683 ClientVolume* volume = _GetVolume(request->volumeID); 684 if (!volume) 685 result = B_BAD_VALUE; 686 ClientVolumePutter volumePutter(this, volume); 687 688 VolumeManagerLocker managerLocker; 689 690 // get the node 691 Node* node = NULL; 692 if (result == B_OK) { 693 node = volume->GetNode(request->nodeID); 694 if (!node) 695 result = B_ENTRY_NOT_FOUND; 696 } 697 698 // check permissions 699 if (result == B_OK) { 700 if (!volume->GetNodePermissions(node).ImpliesWritePermission()) 701 result = B_PERMISSION_DENIED; 702 } 703 704 // get the path 705 Path path; 706 if (result == B_OK) 707 result = node->GetPath(&path); 708 709 // write the stat 710 uint32 mask = request->mask; 711 // size 712 if (result == B_OK && (mask & WSTAT_SIZE)) { 713 if (truncate(path.GetPath(), request->nodeInfo.st.st_size) < 0) 714 result = errno; 715 } 716 // mode 717 if (result == B_OK && (mask & WSTAT_MODE)) { 718 if (chmod(path.GetPath(), request->nodeInfo.st.st_mode) < 0) 719 result = errno; 720 } 721 // mtime 722 if (result == B_OK && (mask & (WSTAT_ATIME | WSTAT_MTIME))) { 723 utimbuf buffer; 724 buffer.actime = (mask & WSTAT_ATIME) 725 ? request->nodeInfo.st.st_atime 726 : node->GetStat().st_atime; 727 buffer.modtime = (mask & WSTAT_MTIME) 728 ? request->nodeInfo.st.st_mtime 729 : node->GetStat().st_mtime; 730 if (utime(path.GetPath(), &buffer) < 0) 731 result = errno; 732 } 733 // ignore WSTAT_CRTIME, WSTAT_UID, WSTAT_GID for the time being 734 735 // prepare the reply 736 WriteStatReply reply; 737 // update the node stat 738 reply.nodeInfoValid = false; 739 if (node) { 740 if (node->UpdateStat() == B_OK) { 741 _GetNodeInfo(node, &reply.nodeInfo); 742 reply.nodeInfoValid = true; 743 } 744 } 745 746 managerLocker.Unlock(); 747 748 // send the reply 749 reply.error = result; 750 return GetChannel()->SendRequest(&reply); 751 } 752 753 // VisitCreateFileRequest 754 status_t 755 ClientConnection::VisitCreateFileRequest(CreateFileRequest* request) 756 { 757 ConnectionReference connectionReference(this); 758 if (!connectionReference.IsValid()) 759 return B_OK; 760 761 // get the volume 762 status_t result = B_OK; 763 ClientVolume* volume = _GetVolume(request->volumeID); 764 if (!volume) 765 result = B_BAD_VALUE; 766 ClientVolumePutter volumePutter(this, volume); 767 768 VolumeManagerLocker managerLocker; 769 770 // get the directory 771 Directory* directory = NULL; 772 if (result == B_OK) { 773 Node* node = volume->GetNode(request->directoryID); 774 if (node) { 775 directory = dynamic_cast<Directory*>(node); 776 if (!directory) 777 result = B_NOT_A_DIRECTORY; 778 } else 779 result = B_ENTRY_NOT_FOUND; 780 } 781 782 // check permissions 783 int openMode = request->openMode; 784 if (result == B_OK) { 785 if (!volume->GetNodePermissions(directory).ImpliesWritePermission()) 786 result = B_PERMISSION_DENIED; 787 } 788 789 // get the path 790 Path path; 791 if (result == B_OK) { 792 result = directory->GetPath(&path); 793 if (result == B_OK) 794 result = path.Append(request->name.GetString()); 795 } 796 797 // create the file 798 if (result == B_OK) { 799 int fd = -1; 800 result = FDManager::Open(path.GetPath(), 801 openMode | O_CREAT | O_NOTRAVERSE, request->mode, fd); 802 if (result == B_OK) 803 close(fd); 804 } 805 806 // load the new entry 807 Entry* entry = NULL; 808 if (result == B_OK) { 809 VolumeManager* volumeManager = VolumeManager::GetDefault(); 810 811 // if there existed an entry before, we need to delete it, to avoid that 812 // we open the wrong node 813 entry = volumeManager->GetEntry(directory->GetVolumeID(), 814 directory->GetID(), request->name.GetString()); 815 if (entry) 816 volumeManager->DeleteEntry(entry, false); 817 818 // load the new entry 819 entry = NULL; 820 result = volume->LoadEntry(directory, request->name.GetString(), 821 &entry); 822 } 823 824 // open the node 825 FileHandle* handle = NULL; 826 if (result == B_OK) { 827 openMode &= ~(O_CREAT | O_EXCL | O_TRUNC); 828 result = volume->Open(entry->GetNode(), openMode, &handle); 829 } 830 NodeHandleUnlocker handleUnlocker(volume, handle); 831 832 // prepare the reply 833 CreateFileReply reply; 834 if (result == B_OK) { 835 _GetEntryInfo(entry, &reply.entryInfo); 836 reply.cookie = handle->GetCookie(); 837 } 838 839 managerLocker.Unlock(); 840 841 // send the reply 842 reply.error = result; 843 status_t error = GetChannel()->SendRequest(&reply); 844 845 // close the handle, if a send error occurred 846 if (error != B_OK && result == B_OK) 847 volume->Close(handle); 848 849 return error; 850 } 851 852 // VisitOpenRequest 853 status_t 854 ClientConnection::VisitOpenRequest(OpenRequest* request) 855 { 856 ConnectionReference connectionReference(this); 857 if (!connectionReference.IsValid()) 858 return B_OK; 859 860 // get the volume 861 status_t result = B_OK; 862 ClientVolume* volume = _GetVolume(request->volumeID); 863 if (!volume) 864 result = B_BAD_VALUE; 865 ClientVolumePutter volumePutter(this, volume); 866 867 VolumeManagerLocker managerLocker; 868 869 // get the node 870 Node* node = NULL; 871 if (result == B_OK) { 872 node = volume->GetNode(request->nodeID); 873 if (!node) 874 result = B_ENTRY_NOT_FOUND; 875 } 876 877 // check permissions 878 int openMode = request->openMode; 879 if (result == B_OK) { 880 Permissions permissions = volume->GetNodePermissions(node); 881 if ((openMode & O_RWMASK) == O_RDWR) { 882 // read+write: fall back to read/write only, if the other permission 883 // is missing 884 if (!permissions.ImpliesReadPermission()) 885 openMode = (openMode & ~O_RWMASK) | O_WRONLY; 886 else if (!permissions.ImpliesWritePermission()) 887 openMode = (openMode & ~O_RWMASK) | O_RDONLY; 888 } 889 if ((openMode & O_RWMASK) == O_RDONLY) { 890 if (!permissions.ImpliesReadPermission()) 891 result = B_PERMISSION_DENIED; 892 } else if ((openMode & O_RWMASK) == O_WRONLY) { 893 if (!permissions.ImpliesWritePermission()) 894 result = B_PERMISSION_DENIED; 895 } 896 } 897 898 // open the node 899 FileHandle* handle = NULL; 900 if (result == B_OK) 901 result = volume->Open(node, openMode, &handle); 902 NodeHandleUnlocker handleUnlocker(volume, handle); 903 904 // prepare the reply 905 OpenReply reply; 906 if (result == B_OK) { 907 _GetNodeInfo(node, &reply.nodeInfo); 908 reply.cookie = handle->GetCookie(); 909 } 910 911 managerLocker.Unlock(); 912 913 // send the reply 914 reply.error = result; 915 status_t error = GetChannel()->SendRequest(&reply); 916 917 // close the handle, if a send error occurred 918 if (error != B_OK && result == B_OK) 919 volume->Close(handle); 920 921 return error; 922 } 923 924 // VisitCloseRequest 925 status_t 926 ClientConnection::VisitCloseRequest(CloseRequest* request) 927 { 928 ConnectionReference connectionReference(this); 929 if (!connectionReference.IsValid()) 930 return B_OK; 931 932 status_t result = B_OK; 933 934 if (request->volumeID >= 0) { 935 // get the volume 936 ClientVolume* volume = _GetVolume(request->volumeID); 937 if (!volume) 938 SET_ERROR(result, B_BAD_VALUE); 939 ClientVolumePutter volumePutter(this, volume); 940 941 // get the node handle 942 NodeHandle* handle = NULL; 943 if (result == B_OK) 944 SET_ERROR(result, volume->LockNodeHandle(request->cookie, &handle)); 945 NodeHandleUnlocker handleUnlocker(volume, handle); 946 947 VolumeManagerLocker managerLocker; 948 949 // close it 950 if (result == B_OK) 951 SET_ERROR(result, volume->Close(handle)); 952 953 managerLocker.Unlock(); 954 } else { 955 // no volume ID given, so this is a query handle 956 // lock the handle 957 QueryHandle* handle = NULL; 958 SET_ERROR(result, _LockQueryHandle(request->cookie, &handle)); 959 QueryHandleUnlocker handleUnlocker(this, handle); 960 961 // close it 962 if (result == B_OK) 963 SET_ERROR(result, _CloseQuery(handle)); 964 } 965 966 // send the reply 967 CloseReply reply; 968 reply.error = result; 969 return GetChannel()->SendRequest(&reply); 970 } 971 972 // VisitReadRequest 973 status_t 974 ClientConnection::VisitReadRequest(ReadRequest* request) 975 { 976 ConnectionReference connectionReference(this); 977 if (!connectionReference.IsValid()) 978 return B_OK; 979 980 // get the volume 981 status_t result = B_OK; 982 ClientVolume* volume = _GetVolume(request->volumeID); 983 if (!volume) 984 result = B_BAD_VALUE; 985 ClientVolumePutter volumePutter(this, volume); 986 987 // get the node handle 988 NodeHandle* handle = NULL; 989 if (result == B_OK) 990 result = volume->LockNodeHandle(request->cookie, &handle); 991 NodeHandleUnlocker handleUnlocker(volume, handle); 992 993 // check if it is a file handle 994 FileHandle* fileHandle = NULL; 995 if (result == B_OK) { 996 fileHandle = dynamic_cast<FileHandle*>(handle); 997 if (!fileHandle) 998 result = B_BAD_VALUE; 999 } 1000 1001 VolumeManagerLocker managerLocker; 1002 1003 // check read permission 1004 if (result == B_OK) { 1005 Node* node = volume->GetNode(fileHandle->GetNodeRef()); 1006 if (!node || !volume->GetNodePermissions(node).ImpliesReadPermission()) 1007 result = B_PERMISSION_DENIED; 1008 } 1009 1010 managerLocker.Unlock(); 1011 1012 off_t pos = request->pos; 1013 int32 size = request->size; 1014 int32 bufferSize = min(size, kMaxReadBufferSize); 1015 // allocate a buffer 1016 uint8* buffer = NULL; 1017 if (result == B_OK) { 1018 buffer = (uint8*)malloc(bufferSize); 1019 if (!buffer) 1020 result = B_NO_MEMORY; 1021 } 1022 MemoryDeleter bufferDeleter(buffer); 1023 1024 // read as long as there are bytes left to read or an error occurs 1025 bool moreToRead = true; 1026 do { 1027 int32 bytesToRead = min(size, bufferSize); 1028 size_t bytesRead = 0; 1029 if (result == B_OK) 1030 result = fileHandle->Read(pos, buffer, bytesToRead, &bytesRead); 1031 moreToRead = (result == B_OK && bytesRead > 0 1032 && (int32)bytesRead < size); 1033 1034 // prepare the reply 1035 ReadReply reply; 1036 if (result == B_OK) { 1037 reply.pos = pos; 1038 reply.data.SetTo(buffer, bytesRead); 1039 reply.moreToCome = moreToRead; 1040 pos += bytesRead; 1041 size -= bytesRead; 1042 } 1043 1044 // send the reply 1045 reply.error = result; 1046 status_t error = GetChannel()->SendRequest(&reply); 1047 if (error != B_OK) 1048 return error; 1049 } while (moreToRead); 1050 1051 return B_OK; 1052 } 1053 1054 // VisitWriteRequest 1055 status_t 1056 ClientConnection::VisitWriteRequest(WriteRequest* request) 1057 { 1058 ConnectionReference connectionReference(this); 1059 if (!connectionReference.IsValid()) 1060 return B_OK; 1061 1062 // get the volume 1063 status_t result = B_OK; 1064 ClientVolume* volume = _GetVolume(request->volumeID); 1065 if (!volume) 1066 result = B_BAD_VALUE; 1067 ClientVolumePutter volumePutter(this, volume); 1068 1069 // get the node handle 1070 NodeHandle* handle = NULL; 1071 if (result == B_OK) 1072 result = volume->LockNodeHandle(request->cookie, &handle); 1073 NodeHandleUnlocker handleUnlocker(volume, handle); 1074 1075 // check if it is a file handle 1076 FileHandle* fileHandle = NULL; 1077 if (result == B_OK) { 1078 fileHandle = dynamic_cast<FileHandle*>(handle); 1079 if (!fileHandle) 1080 result = B_BAD_VALUE; 1081 } 1082 1083 VolumeManagerLocker managerLocker; 1084 1085 // check read permission 1086 if (result == B_OK) { 1087 Node* node = volume->GetNode(fileHandle->GetNodeRef()); 1088 if (!node || !volume->GetNodePermissions(node).ImpliesWritePermission()) 1089 result = B_PERMISSION_DENIED; 1090 } 1091 1092 managerLocker.Unlock(); 1093 1094 // write until all has been written or an error occurs 1095 off_t pos = request->pos; 1096 int32 size = request->data.GetSize(); 1097 const char* buffer = (const char*)request->data.GetData(); 1098 while (result == B_OK && size > 0) { 1099 size_t bytesWritten; 1100 result = fileHandle->Write(pos, buffer, size, &bytesWritten); 1101 if (result == B_OK) { 1102 pos += bytesWritten; 1103 buffer += bytesWritten; 1104 size -= bytesWritten; 1105 } 1106 } 1107 1108 // prepare the reply 1109 WriteReply reply; 1110 // send the reply 1111 reply.error = result; 1112 return GetChannel()->SendRequest(&reply); 1113 } 1114 1115 // VisitCreateLinkRequest 1116 status_t 1117 ClientConnection::VisitCreateLinkRequest(CreateLinkRequest* request) 1118 { 1119 ConnectionReference connectionReference(this); 1120 if (!connectionReference.IsValid()) 1121 return B_OK; 1122 1123 // get the volume 1124 status_t result = B_OK; 1125 ClientVolume* volume = _GetVolume(request->volumeID); 1126 if (!volume) 1127 result = B_BAD_VALUE; 1128 ClientVolumePutter volumePutter(this, volume); 1129 1130 VolumeManagerLocker managerLocker; 1131 1132 // get the target node 1133 Node* node = NULL; 1134 if (result == B_OK) { 1135 node = volume->GetNode(request->nodeID); 1136 if (!node) 1137 result = B_ENTRY_NOT_FOUND; 1138 } 1139 1140 // get the target node path 1141 Path targetPath; 1142 if (result == B_OK) 1143 result = node->GetPath(&targetPath); 1144 1145 // get the directory 1146 Directory* directory = NULL; 1147 if (result == B_OK) { 1148 Node* node = volume->GetNode(request->directoryID); 1149 if (node) { 1150 directory = dynamic_cast<Directory*>(node); 1151 if (!directory) 1152 result = B_NOT_A_DIRECTORY; 1153 } else 1154 result = B_ENTRY_NOT_FOUND; 1155 } 1156 1157 // check permissions 1158 if (result == B_OK) { 1159 if (!volume->GetNodePermissions(directory).ImpliesWritePermission()) 1160 result = B_PERMISSION_DENIED; 1161 } 1162 1163 // get the new entry's path 1164 Path path; 1165 if (result == B_OK) { 1166 result = directory->GetPath(&path); 1167 if (result == B_OK) 1168 result = path.Append(request->name.GetString()); 1169 } 1170 1171 managerLocker.Unlock(); 1172 1173 // create the link 1174 if (result == B_OK) { 1175 if (link(targetPath.GetPath(), path.GetPath()) < 0) 1176 result = errno; 1177 } 1178 1179 // prepare the reply 1180 CreateSymlinkReply reply; 1181 // send the reply 1182 reply.error = result; 1183 return GetChannel()->SendRequest(&reply); 1184 } 1185 1186 // VisitUnlinkRequest 1187 status_t 1188 ClientConnection::VisitUnlinkRequest(UnlinkRequest* request) 1189 { 1190 ConnectionReference connectionReference(this); 1191 if (!connectionReference.IsValid()) 1192 return B_OK; 1193 1194 // get the volume 1195 status_t result = B_OK; 1196 ClientVolume* volume = _GetVolume(request->volumeID); 1197 if (!volume) 1198 result = B_BAD_VALUE; 1199 ClientVolumePutter volumePutter(this, volume); 1200 1201 VolumeManagerLocker managerLocker; 1202 1203 // get the directory 1204 Directory* directory = NULL; 1205 if (result == B_OK) { 1206 Node* node = volume->GetNode(request->directoryID); 1207 if (node) { 1208 directory = dynamic_cast<Directory*>(node); 1209 if (!directory) 1210 result = B_NOT_A_DIRECTORY; 1211 } else 1212 result = B_ENTRY_NOT_FOUND; 1213 } 1214 1215 // check permissions 1216 if (result == B_OK) { 1217 if (!volume->GetNodePermissions(directory).ImpliesWritePermission()) 1218 result = B_PERMISSION_DENIED; 1219 } 1220 1221 // get the entry's path 1222 Path path; 1223 if (result == B_OK) { 1224 result = directory->GetPath(&path); 1225 if (result == B_OK) 1226 result = path.Append(request->name.GetString()); 1227 } 1228 1229 managerLocker.Unlock(); 1230 1231 // remove the entry 1232 if (result == B_OK) { 1233 if (unlink(path.GetPath()) < 0) 1234 result = errno; 1235 } 1236 1237 // prepare the reply 1238 UnlinkReply reply; 1239 // send the reply 1240 reply.error = result; 1241 return GetChannel()->SendRequest(&reply); 1242 } 1243 1244 // VisitCreateSymlinkRequest 1245 status_t 1246 ClientConnection::VisitCreateSymlinkRequest(CreateSymlinkRequest* request) 1247 { 1248 ConnectionReference connectionReference(this); 1249 if (!connectionReference.IsValid()) 1250 return B_OK; 1251 1252 // get the volume 1253 status_t result = B_OK; 1254 ClientVolume* volume = _GetVolume(request->volumeID); 1255 if (!volume) 1256 result = B_BAD_VALUE; 1257 ClientVolumePutter volumePutter(this, volume); 1258 1259 VolumeManagerLocker managerLocker; 1260 1261 // get the directory 1262 Directory* directory = NULL; 1263 if (result == B_OK) { 1264 Node* node = volume->GetNode(request->directoryID); 1265 if (node) { 1266 directory = dynamic_cast<Directory*>(node); 1267 if (!directory) 1268 result = B_NOT_A_DIRECTORY; 1269 } else 1270 result = B_ENTRY_NOT_FOUND; 1271 } 1272 1273 // check permissions 1274 if (result == B_OK) { 1275 if (!volume->GetNodePermissions(directory).ImpliesWritePermission()) 1276 result = B_PERMISSION_DENIED; 1277 } 1278 1279 // get the new entry's path 1280 Path path; 1281 if (result == B_OK) { 1282 result = directory->GetPath(&path); 1283 if (result == B_OK) 1284 result = path.Append(request->name.GetString()); 1285 } 1286 1287 managerLocker.Unlock(); 1288 1289 // create the symlink 1290 if (result == B_OK) { 1291 if (symlink(request->target.GetString(), path.GetPath()) < 0) 1292 result = errno; 1293 } 1294 1295 // prepare the reply 1296 CreateSymlinkReply reply; 1297 // send the reply 1298 reply.error = result; 1299 return GetChannel()->SendRequest(&reply); 1300 } 1301 1302 // VisitReadLinkRequest 1303 status_t 1304 ClientConnection::VisitReadLinkRequest(ReadLinkRequest* request) 1305 { 1306 ConnectionReference connectionReference(this); 1307 if (!connectionReference.IsValid()) 1308 return B_OK; 1309 1310 // get the volume 1311 status_t result = B_OK; 1312 ClientVolume* volume = _GetVolume(request->volumeID); 1313 if (!volume) 1314 result = B_BAD_VALUE; 1315 ClientVolumePutter volumePutter(this, volume); 1316 1317 VolumeManagerLocker managerLocker; 1318 1319 // get the node 1320 Node* node = NULL; 1321 if (result == B_OK) { 1322 node = volume->GetNode(request->nodeID); 1323 if (!node) 1324 result = B_ENTRY_NOT_FOUND; 1325 } 1326 1327 int32 bufferSize = request->maxSize; 1328 1329 // check read permission 1330 if (result == B_OK) { 1331 if (!volume->GetNodePermissions(node).ImpliesReadPermission()) 1332 result = B_PERMISSION_DENIED; 1333 } 1334 1335 // allocate a buffer 1336 void* buffer = NULL; 1337 if (result == B_OK) { 1338 if (bufferSize < 0 || bufferSize > kMaxSaneReadLinkSize) 1339 result = B_BAD_DATA; 1340 } 1341 if (result == B_OK) { 1342 buffer = malloc(bufferSize); 1343 if (!buffer) 1344 result = B_NO_MEMORY; 1345 } 1346 MemoryDeleter bufferDeleter(buffer); 1347 1348 // read the link and prepare the reply 1349 ReadLinkReply reply; 1350 int32 bytesRead = 0; 1351 if (result == B_OK) 1352 result = node->ReadSymlink((char*)buffer, bufferSize, &bytesRead); 1353 if (result == B_OK) { 1354 reply.data.SetTo(buffer, bytesRead); 1355 _GetNodeInfo(node, &reply.nodeInfo); 1356 } 1357 1358 managerLocker.Unlock(); 1359 1360 // send the reply 1361 reply.error = result; 1362 return GetChannel()->SendRequest(&reply); 1363 } 1364 1365 // VisitRenameRequest 1366 status_t 1367 ClientConnection::VisitRenameRequest(RenameRequest* request) 1368 { 1369 ConnectionReference connectionReference(this); 1370 if (!connectionReference.IsValid()) 1371 return B_OK; 1372 1373 // get the volume 1374 status_t result = B_OK; 1375 ClientVolume* volume = _GetVolume(request->volumeID); 1376 if (!volume) 1377 result = B_BAD_VALUE; 1378 ClientVolumePutter volumePutter(this, volume); 1379 1380 VolumeManagerLocker managerLocker; 1381 1382 // get the new directory 1383 Directory* newDirectory = NULL; 1384 if (result == B_OK) { 1385 Node* node = volume->GetNode(request->newDirectoryID); 1386 if (node) { 1387 newDirectory = dynamic_cast<Directory*>(node); 1388 if (!newDirectory) 1389 result = B_NOT_A_DIRECTORY; 1390 } else 1391 result = B_ENTRY_NOT_FOUND; 1392 } 1393 1394 // check permissions 1395 if (result == B_OK) { 1396 if (!volume->GetNodePermissions(newDirectory).ImpliesWritePermission()) 1397 result = B_PERMISSION_DENIED; 1398 } 1399 1400 // get the new path 1401 Path newPath; 1402 if (result == B_OK) { 1403 result = newDirectory->GetPath(&newPath); 1404 if (result == B_OK) 1405 result = newPath.Append(request->newName.GetString()); 1406 } 1407 1408 // get the old directory 1409 Directory* oldDirectory = NULL; 1410 if (result == B_OK) { 1411 Node* node = volume->GetNode(request->oldDirectoryID); 1412 if (node) { 1413 oldDirectory = dynamic_cast<Directory*>(node); 1414 if (!oldDirectory) 1415 result = B_NOT_A_DIRECTORY; 1416 } else 1417 result = B_ENTRY_NOT_FOUND; 1418 } 1419 1420 // check permissions 1421 if (result == B_OK) { 1422 if (!volume->GetNodePermissions(oldDirectory).ImpliesWritePermission()) 1423 result = B_PERMISSION_DENIED; 1424 } 1425 1426 // get the new path 1427 Path oldPath; 1428 if (result == B_OK) { 1429 result = oldDirectory->GetPath(&oldPath); 1430 if (result == B_OK) 1431 result = oldPath.Append(request->oldName.GetString()); 1432 } 1433 1434 managerLocker.Unlock(); 1435 1436 // rename the entry 1437 if (result == B_OK) { 1438 if (rename(oldPath.GetPath(), newPath.GetPath()) < 0) 1439 result = errno; 1440 } 1441 1442 // prepare the reply 1443 RenameReply reply; 1444 // send the reply 1445 reply.error = result; 1446 return GetChannel()->SendRequest(&reply); 1447 } 1448 1449 // VisitMakeDirRequest 1450 status_t 1451 ClientConnection::VisitMakeDirRequest(MakeDirRequest* request) 1452 { 1453 ConnectionReference connectionReference(this); 1454 if (!connectionReference.IsValid()) 1455 return B_OK; 1456 1457 // get the volume 1458 status_t result = B_OK; 1459 ClientVolume* volume = _GetVolume(request->volumeID); 1460 if (!volume) 1461 result = B_BAD_VALUE; 1462 ClientVolumePutter volumePutter(this, volume); 1463 1464 VolumeManagerLocker managerLocker; 1465 1466 // get the directory 1467 Directory* directory = NULL; 1468 if (result == B_OK) { 1469 Node* node = volume->GetNode(request->directoryID); 1470 if (node) { 1471 directory = dynamic_cast<Directory*>(node); 1472 if (!directory) 1473 result = B_NOT_A_DIRECTORY; 1474 } else 1475 result = B_ENTRY_NOT_FOUND; 1476 } 1477 1478 // check permissions 1479 if (result == B_OK) { 1480 if (!volume->GetNodePermissions(directory).ImpliesWritePermission()) 1481 result = B_PERMISSION_DENIED; 1482 } 1483 1484 // get the path 1485 Path path; 1486 if (result == B_OK) { 1487 result = directory->GetPath(&path); 1488 if (result == B_OK) 1489 result = path.Append(request->name.GetString()); 1490 } 1491 1492 managerLocker.Unlock(); 1493 1494 // create the directory 1495 if (result == B_OK) { 1496 if (mkdir(path.GetPath(), request->mode) < 0) 1497 result = errno; 1498 } 1499 1500 // prepare the reply 1501 MakeDirReply reply; 1502 // send the reply 1503 reply.error = result; 1504 return GetChannel()->SendRequest(&reply); 1505 } 1506 1507 // VisitRemoveDirRequest 1508 status_t 1509 ClientConnection::VisitRemoveDirRequest(RemoveDirRequest* request) 1510 { 1511 ConnectionReference connectionReference(this); 1512 if (!connectionReference.IsValid()) 1513 return B_OK; 1514 1515 // get the volume 1516 status_t result = B_OK; 1517 ClientVolume* volume = _GetVolume(request->volumeID); 1518 if (!volume) 1519 result = B_BAD_VALUE; 1520 ClientVolumePutter volumePutter(this, volume); 1521 1522 VolumeManagerLocker managerLocker; 1523 1524 // get the directory 1525 Directory* directory = NULL; 1526 if (result == B_OK) { 1527 Node* node = volume->GetNode(request->directoryID); 1528 if (node) { 1529 directory = dynamic_cast<Directory*>(node); 1530 if (!directory) 1531 result = B_NOT_A_DIRECTORY; 1532 } else 1533 result = B_ENTRY_NOT_FOUND; 1534 } 1535 1536 // check permissions 1537 if (result == B_OK) { 1538 if (!volume->GetNodePermissions(directory).ImpliesWritePermission()) 1539 result = B_PERMISSION_DENIED; 1540 } 1541 1542 // get the path 1543 Path path; 1544 if (result == B_OK) { 1545 result = directory->GetPath(&path); 1546 if (result == B_OK) 1547 result = path.Append(request->name.GetString()); 1548 } 1549 1550 managerLocker.Unlock(); 1551 1552 // remove the directory 1553 if (result == B_OK) { 1554 if (rmdir(path.GetPath()) < 0) 1555 result = errno; 1556 } 1557 1558 // prepare the reply 1559 RemoveDirReply reply; 1560 // send the reply 1561 reply.error = result; 1562 return GetChannel()->SendRequest(&reply); 1563 } 1564 1565 // VisitOpenDirRequest 1566 status_t 1567 ClientConnection::VisitOpenDirRequest(OpenDirRequest* request) 1568 { 1569 ConnectionReference connectionReference(this); 1570 if (!connectionReference.IsValid()) 1571 return B_OK; 1572 1573 // get the volume 1574 status_t result = B_OK; 1575 ClientVolume* volume = _GetVolume(request->volumeID); 1576 if (!volume) 1577 result = B_BAD_VALUE; 1578 ClientVolumePutter volumePutter(this, volume); 1579 1580 VolumeManagerLocker managerLocker; 1581 1582 // get the directory 1583 Directory* directory = NULL; 1584 if (result == B_OK) { 1585 Node* node = volume->GetNode(request->nodeID); 1586 if (node) { 1587 directory = dynamic_cast<Directory*>(node); 1588 if (!directory) 1589 result = B_NOT_A_DIRECTORY; 1590 } else 1591 result = B_ENTRY_NOT_FOUND; 1592 } 1593 1594 // check permission 1595 if (result == B_OK) { 1596 if (!volume->GetNodePermissions(directory).ImpliesReadDirPermission()) 1597 result = B_PERMISSION_DENIED; 1598 } 1599 1600 // open the directory 1601 DirIterator* handle = NULL; 1602 if (result == B_OK) 1603 result = volume->OpenDir(directory, &handle); 1604 NodeHandleUnlocker handleUnlocker(volume, handle); 1605 1606 // prepare the reply 1607 OpenDirReply reply; 1608 if (result == B_OK) { 1609 _GetNodeInfo(directory, &reply.nodeInfo); 1610 reply.cookie = handle->GetCookie(); 1611 } else { 1612 if (directory != NULL) { 1613 PRINT("OpenDir() failed: client volume: %" B_PRId32 ", " 1614 "node: (%" B_PRIdDEV ", %" B_PRIdINO ")\n", 1615 volume->GetID(), directory->GetVolumeID(), directory->GetID()); 1616 } 1617 } 1618 1619 managerLocker.Unlock(); 1620 1621 // send the reply 1622 reply.error = result; 1623 status_t error = GetChannel()->SendRequest(&reply); 1624 1625 // close the handle, if a send error occurred 1626 if (error != B_OK && result == B_OK) 1627 volume->Close(handle); 1628 1629 return error; 1630 } 1631 1632 // VisitReadDirRequest 1633 status_t 1634 ClientConnection::VisitReadDirRequest(ReadDirRequest* request) 1635 { 1636 ConnectionReference connectionReference(this); 1637 if (!connectionReference.IsValid()) 1638 return B_OK; 1639 1640 // get the volume 1641 status_t result = B_OK; 1642 ClientVolume* volume = _GetVolume(request->volumeID); 1643 if (!volume) 1644 result = B_BAD_VALUE; 1645 ClientVolumePutter volumePutter(this, volume); 1646 1647 // get the node handle 1648 NodeHandle* handle = NULL; 1649 if (result == B_OK) 1650 result = volume->LockNodeHandle(request->cookie, &handle); 1651 NodeHandleUnlocker handleUnlocker(volume, handle); 1652 1653 // check if it is a directory iterator 1654 DirIterator* iterator = NULL; 1655 if (result == B_OK) { 1656 iterator = dynamic_cast<DirIterator*>(handle); 1657 if (!iterator) 1658 result = B_BAD_VALUE; 1659 } 1660 1661 VolumeManagerLocker managerLocker; 1662 1663 // get the directory 1664 Directory* directory = NULL; 1665 if (result == B_OK) { 1666 Node* node = volume->GetNode(iterator->GetNodeRef()); 1667 if (node) { 1668 directory = dynamic_cast<Directory*>(node); 1669 if (!directory) 1670 result = B_NOT_A_DIRECTORY; 1671 } else 1672 result = B_ENTRY_NOT_FOUND; 1673 } 1674 1675 // check read permission 1676 if (result == B_OK) { 1677 if (!volume->GetNodePermissions(directory).ImpliesReadDirPermission()) 1678 result = B_PERMISSION_DENIED; 1679 } 1680 1681 if (result == B_OK) { 1682 PRINT("ReadDir: (%" B_PRId32 ", %" B_PRIdINO ")\n", request->volumeID, 1683 directory->GetID()); 1684 } 1685 1686 // rewind, if requested 1687 if (result == B_OK && request->rewind) 1688 iterator->Rewind(); 1689 1690 // read the directory 1691 bool done = false; 1692 ReadDirReply reply; 1693 int32 toRead = request->count; 1694 while (result == B_OK && toRead > 0) { 1695 // get the next entry 1696 Entry* entry = iterator->NextEntry(); 1697 if (!entry) { 1698 done = true; 1699 break; 1700 } 1701 1702 // get and add an entry info 1703 EntryInfo entryInfo; 1704 _GetEntryInfo(entry, &entryInfo); 1705 result = reply.entryInfos.Append(entryInfo); 1706 1707 toRead--; 1708 } 1709 1710 reply.revision = VolumeManager::GetDefault()->GetRevision(); 1711 1712 //PRINT(("ReadDir: (%lld) -> (%lx, %ld, dir: %lld, node: %lld, `%s')\n", 1713 //directoryID, reply.error, reply.entryInfos.CountElements(), 1714 //reply.entryInfo.directoryID, 1715 //reply.entryInfo.nodeID, reply.entryInfo.name.GetString())); 1716 if (directory != NULL) { 1717 PRINT("ReadDir done: volume: %" B_PRId32 ", " 1718 "(%" B_PRIdDEV ", %" B_PRIdINO ") -> " 1719 "(%" B_PRIx32 ", %" B_PRId32 ")\n", 1720 volume->GetID(), directory->GetVolumeID(), directory->GetID(), 1721 result, reply.entryInfos.CountElements()); 1722 } 1723 1724 managerLocker.Unlock(); 1725 1726 // send the reply 1727 reply.error = result; 1728 reply.done = (result != B_OK || done); 1729 return GetChannel()->SendRequest(&reply); 1730 } 1731 1732 // VisitWalkRequest 1733 status_t 1734 ClientConnection::VisitWalkRequest(WalkRequest* request) 1735 { 1736 ConnectionReference connectionReference(this); 1737 if (!connectionReference.IsValid()) 1738 return B_OK; 1739 1740 // get the volume 1741 status_t result = B_OK; 1742 ClientVolume* volume = _GetVolume(request->volumeID); 1743 if (!volume) 1744 result = B_BAD_VALUE; 1745 ClientVolumePutter volumePutter(this, volume); 1746 1747 VolumeManagerLocker managerLocker; 1748 1749 // get the directory 1750 Directory* directory = NULL; 1751 if (result == B_OK) { 1752 Node* node = volume->GetNode(request->nodeID); 1753 if (node) { 1754 directory = dynamic_cast<Directory*>(node); 1755 if (!directory) 1756 result = B_NOT_A_DIRECTORY; 1757 } else 1758 result = B_ENTRY_NOT_FOUND; 1759 } 1760 1761 // check permission 1762 if (result == B_OK) { 1763 if (!volume->GetNodePermissions(directory) 1764 .ImpliesResolveDirEntryPermission()) { 1765 result = B_PERMISSION_DENIED; 1766 } 1767 } 1768 1769 WalkReply reply; 1770 char linkPath[B_PATH_NAME_LENGTH]; 1771 if (result == B_OK) { 1772 // load the entry 1773 Entry* entry; 1774 result = volume->LoadEntry(directory, request->name.GetString(), 1775 &entry); 1776 1777 // fill in the reply 1778 if (result == B_OK) { 1779 _GetEntryInfo(entry, &reply.entryInfo); 1780 1781 // resolve a symlink, if desired 1782 Node* node = entry->GetNode(); 1783 if (request->resolveLink && node->IsSymlink()) { 1784 result = node->ReadSymlink(linkPath, B_PATH_NAME_LENGTH); 1785 if (result == B_OK) 1786 reply.linkPath.SetTo(linkPath); 1787 } 1788 } 1789 } 1790 1791 managerLocker.Unlock(); 1792 1793 // send the reply 1794 reply.error = result; 1795 PRINT("Walk: (%" B_PRIdDEV ", %" B_PRIdINO ", `%s') -> " 1796 "(%" B_PRIx32 ", (%" B_PRIdDEV ", %" B_PRIdINO "), `%s')\n", 1797 request->nodeID.volumeID, request->nodeID.nodeID, 1798 request->name.GetString(), result, 1799 reply.entryInfo.nodeInfo.st.st_dev, 1800 reply.entryInfo.nodeInfo.st.st_ino, reply.linkPath.GetString()); 1801 return GetChannel()->SendRequest(&reply); 1802 } 1803 1804 // VisitMultiWalkRequest 1805 status_t 1806 ClientConnection::VisitMultiWalkRequest(MultiWalkRequest* request) 1807 { 1808 ConnectionReference connectionReference(this); 1809 if (!connectionReference.IsValid()) 1810 return B_OK; 1811 1812 // get the volume 1813 status_t result = B_OK; 1814 ClientVolume* volume = _GetVolume(request->volumeID); 1815 if (!volume) 1816 result = B_BAD_VALUE; 1817 ClientVolumePutter volumePutter(this, volume); 1818 1819 VolumeManagerLocker managerLocker; 1820 1821 // get the directory 1822 Directory* directory = NULL; 1823 if (result == B_OK) { 1824 Node* node = volume->GetNode(request->nodeID); 1825 if (node) { 1826 directory = dynamic_cast<Directory*>(node); 1827 if (!directory) 1828 result = B_NOT_A_DIRECTORY; 1829 } else 1830 result = B_ENTRY_NOT_FOUND; 1831 } 1832 1833 // check permission 1834 if (result == B_OK) { 1835 if (!volume->GetNodePermissions(directory) 1836 .ImpliesResolveDirEntryPermission()) { 1837 result = B_PERMISSION_DENIED; 1838 } 1839 } 1840 1841 MultiWalkReply reply; 1842 StringData* names = request->names.GetElements(); 1843 int32 count = request->names.CountElements(); 1844 for (int32 i = 0; result == B_OK && i < count; i++) { 1845 // load the entry 1846 Entry* entry; 1847 if (volume->LoadEntry(directory, names[i].GetString(), &entry) 1848 == B_OK) { 1849 // add an entry info 1850 EntryInfo entryInfo; 1851 _GetEntryInfo(entry, &entryInfo); 1852 1853 // append the info 1854 result = reply.entryInfos.Append(entryInfo); 1855 } 1856 } 1857 1858 managerLocker.Unlock(); 1859 1860 // send the reply 1861 reply.error = result; 1862 PRINT("MultiWalk: (%" B_PRIdDEV ", %" B_PRIdINO ", %" B_PRId32 ") -> " 1863 "(%" B_PRIx32 ", %" B_PRId32 ")\n", 1864 request->nodeID.volumeID, request->nodeID.nodeID, count, 1865 result, reply.entryInfos.CountElements()); 1866 return GetChannel()->SendRequest(&reply); 1867 } 1868 1869 // VisitOpenAttrDirRequest 1870 status_t 1871 ClientConnection::VisitOpenAttrDirRequest(OpenAttrDirRequest* request) 1872 { 1873 ConnectionReference connectionReference(this); 1874 if (!connectionReference.IsValid()) 1875 return B_OK; 1876 1877 // get the volume 1878 status_t result = B_OK; 1879 ClientVolume* volume = _GetVolume(request->volumeID); 1880 if (!volume) 1881 result = B_BAD_VALUE; 1882 ClientVolumePutter volumePutter(this, volume); 1883 1884 VolumeManagerLocker managerLocker; 1885 1886 // get the node 1887 Node* node = NULL; 1888 if (result == B_OK) { 1889 node = volume->GetNode(request->nodeID); 1890 if (!node) 1891 result = B_ENTRY_NOT_FOUND; 1892 } 1893 1894 // check permission 1895 if (result == B_OK) { 1896 if (!volume->GetNodePermissions(node).ImpliesReadPermission()) 1897 result = B_PERMISSION_DENIED; 1898 } 1899 1900 // load/cache the attribute directory 1901 bool attrDirCached = (node->LoadAttrDir() == B_OK); 1902 1903 // open the attribute directory, if caching it failed 1904 AttrDirIterator* handle = NULL; 1905 if (result == B_OK && !attrDirCached) 1906 result = volume->OpenAttrDir(node, &handle); 1907 NodeHandleUnlocker handleUnlocker(volume, handle); 1908 1909 // prepare the reply 1910 OpenAttrDirReply reply; 1911 if (result == B_OK) { 1912 if (handle) { 1913 reply.cookie = handle->GetCookie(); 1914 } else { 1915 // the attribute directory is cached 1916 reply.cookie = -1; 1917 result = _GetAttrDirInfo(request, node, &reply.attrDirInfo); 1918 } 1919 } 1920 1921 managerLocker.Unlock(); 1922 1923 // send the reply 1924 reply.error = result; 1925 status_t error = GetChannel()->SendRequest(&reply); 1926 1927 // close the handle, if a send error occurred 1928 if (error != B_OK && result == B_OK && handle) 1929 volume->Close(handle); 1930 1931 return error; 1932 } 1933 1934 // VisitReadAttrDirRequest 1935 status_t 1936 ClientConnection::VisitReadAttrDirRequest(ReadAttrDirRequest* request) 1937 { 1938 ConnectionReference connectionReference(this); 1939 if (!connectionReference.IsValid()) 1940 return B_OK; 1941 1942 // get the volume 1943 status_t result = B_OK; 1944 ClientVolume* volume = _GetVolume(request->volumeID); 1945 if (!volume) 1946 result = B_BAD_VALUE; 1947 ClientVolumePutter volumePutter(this, volume); 1948 1949 // get the node handle 1950 NodeHandle* handle = NULL; 1951 if (result == B_OK) 1952 result = volume->LockNodeHandle(request->cookie, &handle); 1953 NodeHandleUnlocker handleUnlocker(volume, handle); 1954 1955 // check if it is a attribute directory iterator 1956 AttrDirIterator* iterator = NULL; 1957 if (result == B_OK) { 1958 iterator = dynamic_cast<AttrDirIterator*>(handle); 1959 if (!iterator) 1960 result = B_BAD_VALUE; 1961 } 1962 1963 VolumeManagerLocker managerLocker; 1964 1965 // get the node 1966 Node* node = NULL; 1967 if (result == B_OK) { 1968 node = volume->GetNode(iterator->GetNodeRef()); 1969 if (!node) 1970 result = B_ENTRY_NOT_FOUND; 1971 } 1972 1973 // check read permission (we already checked when opening, but anyway...) 1974 if (result == B_OK) { 1975 if (!volume->GetNodePermissions(node).ImpliesReadPermission()) 1976 result = B_PERMISSION_DENIED; 1977 } 1978 1979 managerLocker.Unlock(); 1980 1981 // read the attribute directory 1982 uint8 buffer[offsetof(struct dirent, d_name) + B_FILE_NAME_LENGTH]; 1983 struct dirent* dirEntry = (struct dirent*)buffer; 1984 int32 countRead = 0; 1985 bool done = true; 1986 if (result == B_OK) { 1987 if (request->rewind) 1988 result = iterator->RewindDir(); 1989 if (result == B_OK) { 1990 result = iterator->ReadDir(dirEntry, sizeof(buffer), 1, 1991 &countRead, &done); 1992 } 1993 } 1994 1995 // prepare the reply 1996 ReadAttrDirReply reply; 1997 reply.name.SetTo(dirEntry->d_name); 1998 1999 // send the reply 2000 reply.error = result; 2001 reply.count = countRead; 2002 return GetChannel()->SendRequest(&reply); 2003 } 2004 2005 // VisitReadAttrRequest 2006 status_t 2007 ClientConnection::VisitReadAttrRequest(ReadAttrRequest* request) 2008 { 2009 ConnectionReference connectionReference(this); 2010 if (!connectionReference.IsValid()) 2011 return B_OK; 2012 2013 // get the volume 2014 status_t result = B_OK; 2015 ClientVolume* volume = _GetVolume(request->volumeID); 2016 if (!volume) 2017 result = B_BAD_VALUE; 2018 ClientVolumePutter volumePutter(this, volume); 2019 2020 VolumeManagerLocker managerLocker; 2021 2022 // get the node 2023 Node* node = NULL; 2024 if (result == B_OK) { 2025 node = volume->GetNode(request->nodeID); 2026 if (!node) 2027 result = B_ENTRY_NOT_FOUND; 2028 } 2029 2030 // check read permission 2031 if (result == B_OK) { 2032 if (!volume->GetNodePermissions(node).ImpliesReadPermission()) 2033 result = B_PERMISSION_DENIED; 2034 } 2035 2036 // open the node 2037 FileHandle* handle = NULL; 2038 if (result == B_OK) 2039 result = volume->Open(node, O_RDONLY, &handle); 2040 NodeHandleUnlocker handleUnlocker(volume, handle); 2041 2042 managerLocker.Unlock(); 2043 2044 // read the attribute 2045 if (result == B_OK) { 2046 // Due to a bug in BFS the `pos' is ignored. This means that the loop 2047 // below won't work if the attribute is non-empty and the buffer is 2048 // larger than the attribute, because the we would again and again 2049 // read the beginning of the attribute until the buffer is full. 2050 // Hence we first get an attr_info and don't try to read more than 2051 // the size of the attribute. 2052 attr_info info; 2053 result = handle->StatAttr(request->name.GetString(), &info); 2054 off_t originalPos = max(request->pos, (off_t)0); 2055 int32 originalSize = max(request->size, (int32)0); 2056 off_t pos = originalPos; 2057 int32 size = originalSize; 2058 type_code type = B_SWAP_INT32(request->type); 2059 bool convert = false; 2060 2061 if (result == B_OK) { 2062 originalSize = min((off_t)originalSize, max((off_t)0, info.size - pos)); 2063 size = originalSize; 2064 2065 // deal with inverse endianess clients 2066 if (fInverseClientEndianess) { 2067 convert = _KnownAttributeType(info.type); 2068 if (convert) { 2069 // read the whole attribute 2070 pos = 0; 2071 size = info.size; 2072 } else 2073 type = B_SWAP_INT32(request->type); 2074 } 2075 } 2076 int32 bufferSize = min(size, kMaxReadBufferSize); 2077 2078 // allocate a buffer 2079 uint8* buffer = NULL; 2080 if (result == B_OK) { 2081 buffer = (uint8*)malloc(bufferSize); 2082 if (!buffer) 2083 result = B_NO_MEMORY; 2084 } 2085 MemoryDeleter bufferDeleter(buffer); 2086 2087 if (convert) { 2088 // read the whole attribute and convert it 2089 if (result == B_OK) { 2090 // read 2091 size_t bytesRead = 0; 2092 result = handle->ReadAttr(request->name.GetString(), 2093 type, 0, buffer, size, &bytesRead); 2094 if (result == B_OK && (int32)bytesRead != size) 2095 result = B_ERROR; 2096 2097 // convert 2098 if (result == B_OK) 2099 _ConvertAttribute(info, buffer); 2100 } 2101 2102 // prepare the reply 2103 ReadAttrReply reply; 2104 if (result == B_OK) { 2105 reply.pos = originalPos; 2106 reply.data.SetTo(buffer + originalPos, originalSize); 2107 reply.moreToCome = false; 2108 } 2109 2110 // send the reply 2111 reply.error = result; 2112 status_t error = GetChannel()->SendRequest(&reply); 2113 if (error != B_OK) 2114 return error; 2115 } else { 2116 // read as long as there are bytes left to read or an error occurs 2117 bool moreToRead = true; 2118 do { 2119 int32 bytesToRead = min(size, bufferSize); 2120 size_t bytesRead = 0; 2121 if (result == B_OK) { 2122 result = handle->ReadAttr(request->name.GetString(), 2123 request->type, pos, buffer, bytesToRead, &bytesRead); 2124 } 2125 moreToRead = (result == B_OK && bytesRead > 0 2126 && (int32)bytesRead < size); 2127 2128 // prepare the reply 2129 ReadAttrReply reply; 2130 if (result == B_OK) { 2131 reply.pos = pos; 2132 reply.data.SetTo(buffer, bytesRead); 2133 reply.moreToCome = moreToRead; 2134 pos += bytesRead; 2135 size -= bytesRead; 2136 } 2137 2138 // send the reply 2139 reply.error = result; 2140 status_t error = GetChannel()->SendRequest(&reply); 2141 if (error != B_OK) 2142 return error; 2143 } while (moreToRead); 2144 } 2145 2146 // close the handle 2147 volume->Close(handle); 2148 } else { 2149 // opening the node failed (or something even earlier): send an error 2150 // reply 2151 ReadAttrReply reply; 2152 reply.error = result; 2153 status_t error = GetChannel()->SendRequest(&reply); 2154 if (error != B_OK) 2155 return error; 2156 } 2157 2158 return B_OK; 2159 } 2160 2161 // VisitWriteAttrRequest 2162 status_t 2163 ClientConnection::VisitWriteAttrRequest(WriteAttrRequest* request) 2164 { 2165 ConnectionReference connectionReference(this); 2166 if (!connectionReference.IsValid()) 2167 return B_OK; 2168 2169 // get the volume 2170 status_t result = B_OK; 2171 ClientVolume* volume = _GetVolume(request->volumeID); 2172 if (!volume) 2173 result = B_BAD_VALUE; 2174 ClientVolumePutter volumePutter(this, volume); 2175 2176 VolumeManagerLocker managerLocker; 2177 2178 // get the node 2179 Node* node = NULL; 2180 if (result == B_OK) { 2181 node = volume->GetNode(request->nodeID); 2182 if (!node) 2183 result = B_ENTRY_NOT_FOUND; 2184 } 2185 2186 // check read permission 2187 if (result == B_OK) { 2188 if (!volume->GetNodePermissions(node).ImpliesWritePermission()) 2189 result = B_PERMISSION_DENIED; 2190 } 2191 2192 // open the node 2193 FileHandle* handle = NULL; 2194 if (result == B_OK) 2195 result = volume->Open(node, O_RDWR, &handle); 2196 NodeHandleUnlocker handleUnlocker(volume, handle); 2197 2198 managerLocker.Unlock(); 2199 2200 if (result == B_OK) { 2201 off_t pos = max(request->pos, (off_t)0); 2202 int32 size = request->data.GetSize(); 2203 type_code type = request->type; 2204 char* buffer = (char*)request->data.GetData(); 2205 2206 // convert the data, if necessary 2207 if (fInverseClientEndianess) { 2208 if (_KnownAttributeType(type)) { 2209 if (pos != 0) { 2210 WARN("WriteAttr(): WARNING: Need to convert attribute " 2211 "endianess, but position is not 0: attribute: %s, " 2212 "pos: %" B_PRIdOFF ", size: %" B_PRId32 "\n", 2213 request->name.GetString(), pos, size); 2214 } 2215 swap_data(type, buffer, size, B_SWAP_ALWAYS); 2216 } else 2217 type = B_SWAP_INT32(type); 2218 } 2219 2220 // write the data 2221 while (result == B_OK && size > 0) { 2222 size_t bytesWritten; 2223 result = handle->WriteAttr(request->name.GetString(), 2224 type, pos, buffer, size, &bytesWritten); 2225 if (result == B_OK) { 2226 pos += bytesWritten; 2227 buffer += bytesWritten; 2228 size -= bytesWritten; 2229 } 2230 } 2231 2232 // close the handle 2233 volume->Close(handle); 2234 } 2235 2236 // prepare the reply 2237 WriteAttrReply reply; 2238 // send the reply 2239 reply.error = result; 2240 return GetChannel()->SendRequest(&reply); 2241 } 2242 2243 // VisitRemoveAttrRequest 2244 status_t 2245 ClientConnection::VisitRemoveAttrRequest(RemoveAttrRequest* request) 2246 { 2247 ConnectionReference connectionReference(this); 2248 if (!connectionReference.IsValid()) 2249 return B_OK; 2250 2251 // get the volume 2252 status_t result = B_OK; 2253 ClientVolume* volume = _GetVolume(request->volumeID); 2254 if (!volume) 2255 result = B_BAD_VALUE; 2256 ClientVolumePutter volumePutter(this, volume); 2257 2258 VolumeManagerLocker managerLocker; 2259 2260 // get the node 2261 Node* node = NULL; 2262 if (result == B_OK) { 2263 node = volume->GetNode(request->nodeID); 2264 if (!node) 2265 result = B_ENTRY_NOT_FOUND; 2266 } 2267 2268 // check read permission 2269 if (result == B_OK) { 2270 if (!volume->GetNodePermissions(node).ImpliesWritePermission()) 2271 result = B_PERMISSION_DENIED; 2272 } 2273 2274 // open the node 2275 FileHandle* handle = NULL; 2276 if (result == B_OK) 2277 result = volume->Open(node, O_RDWR, &handle); 2278 NodeHandleUnlocker handleUnlocker(volume, handle); 2279 2280 managerLocker.Unlock(); 2281 2282 // remove the attribute and close the node 2283 if (result == B_OK) { 2284 result = handle->RemoveAttr(request->name.GetString()); 2285 volume->Close(handle); 2286 } 2287 2288 // send the reply 2289 RemoveAttrReply reply; 2290 reply.error = result; 2291 return GetChannel()->SendRequest(&reply); 2292 } 2293 2294 // VisitRenameAttrRequest 2295 status_t 2296 ClientConnection::VisitRenameAttrRequest(RenameAttrRequest* request) 2297 { 2298 // Not supported, since there's no API function to rename an attribute. 2299 // send the reply 2300 RemoveAttrReply reply; 2301 reply.error = B_UNSUPPORTED; 2302 return GetChannel()->SendRequest(&reply); 2303 } 2304 2305 // VisitStatAttrRequest 2306 status_t 2307 ClientConnection::VisitStatAttrRequest(StatAttrRequest* request) 2308 { 2309 ConnectionReference connectionReference(this); 2310 if (!connectionReference.IsValid()) 2311 return B_OK; 2312 2313 // get the volume 2314 status_t result = B_OK; 2315 ClientVolume* volume = _GetVolume(request->volumeID); 2316 if (!volume) 2317 result = B_BAD_VALUE; 2318 ClientVolumePutter volumePutter(this, volume); 2319 2320 VolumeManagerLocker managerLocker; 2321 2322 // get the node 2323 Node* node = NULL; 2324 if (result == B_OK) { 2325 node = volume->GetNode(request->nodeID); 2326 if (!node) 2327 result = B_ENTRY_NOT_FOUND; 2328 } 2329 2330 // check read permission 2331 if (result == B_OK) { 2332 if (!volume->GetNodePermissions(node).ImpliesReadPermission()) 2333 result = B_PERMISSION_DENIED; 2334 } 2335 2336 // open the node 2337 FileHandle* handle = NULL; 2338 if (result == B_OK) 2339 result = volume->Open(node, O_RDONLY, &handle); 2340 NodeHandleUnlocker handleUnlocker(volume, handle); 2341 2342 managerLocker.Unlock(); 2343 2344 // stat the attribute and close the node 2345 attr_info attrInfo; 2346 StatAttrReply reply; 2347 if (result == B_OK) { 2348 result = handle->StatAttr(request->name.GetString(), &attrInfo); 2349 volume->Close(handle); 2350 } 2351 2352 // set the attribute info 2353 if (result == B_OK) { 2354 result = _GetAttrInfo(request, request->name.GetString(), attrInfo, 2355 NULL, &reply.attrInfo); 2356 } 2357 2358 // send the reply 2359 reply.error = result; 2360 return GetChannel()->SendRequest(&reply); 2361 } 2362 2363 // VisitOpenQueryRequest 2364 status_t 2365 ClientConnection::VisitOpenQueryRequest(OpenQueryRequest* request) 2366 { 2367 ConnectionReference connectionReference(this); 2368 if (!connectionReference.IsValid()) 2369 return B_OK; 2370 2371 VolumeManagerLocker managerLocker; 2372 2373 // open the query 2374 status_t result; 2375 QueryHandle* handle = NULL; 2376 result = _OpenQuery(request->queryString.GetString(), 2377 request->flags, request->port, request->token, &handle); 2378 QueryHandleUnlocker handleUnlocker(this, handle); 2379 2380 // prepare the reply 2381 OpenQueryReply reply; 2382 if (result == B_OK) 2383 reply.cookie = handle->GetCookie(); 2384 2385 managerLocker.Unlock(); 2386 2387 // send the reply 2388 reply.error = result; 2389 status_t error = GetChannel()->SendRequest(&reply); 2390 2391 // close the handle, if a send error occurred 2392 if (error != B_OK && result == B_OK) 2393 _CloseQuery(handle); 2394 2395 return error; 2396 } 2397 2398 // VisitReadQueryRequest 2399 status_t 2400 ClientConnection::VisitReadQueryRequest(ReadQueryRequest* request) 2401 { 2402 ConnectionReference connectionReference(this); 2403 if (!connectionReference.IsValid()) 2404 return B_OK; 2405 2406 // create an array for the IDs of the client volumes a found entry may 2407 // reside on 2408 status_t result = B_OK; 2409 int32 volumeCount = fVolumes->Size(); 2410 int32* volumeIDs = new(std::nothrow) int32[volumeCount]; 2411 if (!volumeIDs) 2412 result = B_NO_MEMORY; 2413 ArrayDeleter<int32> volumeIDsDeleter(volumeIDs); 2414 2415 // get the query handle 2416 QueryHandle* handle = NULL; 2417 if (result == B_OK) 2418 result = _LockQueryHandle(request->cookie, &handle); 2419 QueryHandleUnlocker handleUnlocker(this, handle); 2420 2421 // check if it is a query handle 2422 QueryHandle* queryHandle = NULL; 2423 if (result == B_OK) { 2424 queryHandle = dynamic_cast<QueryHandle*>(handle); 2425 if (!queryHandle) 2426 result = B_BAD_VALUE; 2427 } 2428 2429 // read the query 2430 ReadQueryReply reply; 2431 int32 countRead = 0; 2432 while (result == B_OK) { 2433 uint8 buffer[offsetof(struct dirent, d_name) + B_FILE_NAME_LENGTH]; 2434 struct dirent* dirEntry = (struct dirent*)buffer; 2435 2436 result = queryHandle->ReadDir(dirEntry, 1, &countRead); 2437 if (result != B_OK) 2438 break; 2439 if (countRead == 0) 2440 break; 2441 PRINT(" query entry: %" B_PRIdDEV ", %" B_PRIdINO ", \"%s\"\n", 2442 dirEntry->d_pdev, dirEntry->d_pino, dirEntry->d_name); 2443 2444 VolumeManagerLocker managerLocker; 2445 VolumeManager* volumeManager = VolumeManager::GetDefault(); 2446 2447 // load the entry 2448 Entry* entry = NULL; 2449 result = volumeManager->LoadEntry(dirEntry->d_pdev, 2450 dirEntry->d_pino, dirEntry->d_name, true, &entry); 2451 2452 // if at least one client volume contains the entry, get an entry info 2453 if (result == B_OK) { 2454 HasQueryPermissionClientVolumeFilter filter; 2455 int32 entryVolumeCount = _GetContainingClientVolumes( 2456 entry->GetDirectory(), volumeIDs, volumeCount, &filter); 2457 if (entryVolumeCount > 0) { 2458 // store all the client volume IDs in the reply 2459 for (int32 i = 0; i < entryVolumeCount; i++) { 2460 result = reply.clientVolumeIDs.Append(volumeIDs[i]); 2461 if (result != B_OK) 2462 break; 2463 } 2464 2465 // get an entry info 2466 _GetNodeInfo(entry->GetDirectory(), &reply.dirInfo); 2467 _GetEntryInfo(entry, &reply.entryInfo); 2468 break; 2469 } else 2470 PRINT((" -> no client volumes\n")); 2471 } 2472 2473 // entry is not in the volume: next round... 2474 result = B_OK; 2475 } 2476 2477 // send the reply 2478 reply.error = result; 2479 reply.count = countRead; 2480 PRINT("ReadQuery: (%" B_PRIx32 ", %" B_PRId32 ", " 2481 "dir: (%" B_PRIdDEV ", %" B_PRIdINO "), " 2482 "node: (%" B_PRIdDEV ", %" B_PRIdINO ", `%s')\n", 2483 reply.error, reply.count, 2484 reply.entryInfo.directoryID.volumeID, 2485 reply.entryInfo.directoryID.nodeID, 2486 reply.entryInfo.nodeInfo.st.st_dev, 2487 reply.entryInfo.nodeInfo.st.st_ino, 2488 reply.entryInfo.name.GetString()); 2489 return GetChannel()->SendRequest(&reply); 2490 } 2491 2492 2493 // #pragma mark - 2494 2495 // ProcessNodeMonitoringEvent 2496 void 2497 ClientConnection::ProcessNodeMonitoringEvent(int32 volumeID, 2498 NodeMonitoringEvent* event) 2499 { 2500 // get a connection reference 2501 ConnectionReference connectionReference(this); 2502 if (!connectionReference.IsValid()) 2503 return; 2504 2505 _PushNodeMonitoringEvent(volumeID, event); 2506 } 2507 2508 // CloseNodeMonitoringEventQueue 2509 void 2510 ClientConnection::CloseNodeMonitoringEventQueue() 2511 { 2512 typedef Vector<NodeMonitoringRequest*> RequestVector; 2513 const RequestVector* requests = NULL; 2514 if (fNodeMonitoringEvents->Close(false, &requests) == B_OK) { 2515 for (RequestVector::ConstIterator it = requests->Begin(); 2516 it != requests->End(); 2517 it++) { 2518 delete *it; 2519 } 2520 } 2521 } 2522 2523 2524 // #pragma mark - 2525 2526 // QueryDomainIntersectsWith 2527 bool 2528 ClientConnection::QueryDomainIntersectsWith(Volume* volume) 2529 { 2530 // Iterate through the the client volumes and check whether any one contains 2531 // the supplied volume or its root dir is on the volume. We don't check 2532 // directory inclusion for the latter, since we don't need to query the 2533 // volume, if the client volume is located on a volume mounted somewhere 2534 // under the supplied volume (e.g. the root FS contains everything, but does 2535 // seldomly need to be queried). 2536 VolumeManager* volumeManager = VolumeManager::GetDefault(); 2537 AutoLocker<VolumeMap> volumesLocker(fVolumes); 2538 for (VolumeMap::Iterator it = fVolumes->GetIterator(); it.HasNext();) { 2539 ClientVolume* clientVolume = it.Next().value; 2540 Directory* volumeRoot = volume->GetRootDirectory(); 2541 Directory* clientVolumeRoot = clientVolume->GetRootDirectory(); 2542 if (volumeManager->DirectoryContains(clientVolumeRoot, volumeRoot, true) 2543 || volumeRoot->GetVolumeID() == clientVolumeRoot->GetVolumeID()) { 2544 return true; 2545 } 2546 } 2547 2548 return false; 2549 } 2550 2551 // ProcessQueryEvent 2552 void 2553 ClientConnection::ProcessQueryEvent(NodeMonitoringEvent* event) 2554 { 2555 dev_t volumeID; 2556 ino_t directoryID; 2557 if (event->opcode == B_ENTRY_CREATED) { 2558 // "entry created" event 2559 EntryCreatedEvent* createdEvent 2560 = dynamic_cast<EntryCreatedEvent*>(event); 2561 if (!createdEvent) 2562 return; 2563 volumeID = createdEvent->volumeID; 2564 directoryID = createdEvent->directoryID; 2565 2566 } else if (event->opcode == B_ENTRY_REMOVED) { 2567 // "entry removed" event 2568 EntryRemovedEvent* removedEvent 2569 = dynamic_cast<EntryRemovedEvent*>(event); 2570 if (!removedEvent) 2571 return; 2572 volumeID = removedEvent->volumeID; 2573 directoryID = removedEvent->directoryID; 2574 2575 } else { 2576 // We only support "entry created" and "entry removed" query events. 2577 // "entry moved" is split by the volume manager into those. 2578 ERROR("Ignoring unexpected query event: opcode: 0x%" B_PRIx32 "\n", 2579 event->opcode); 2580 return; 2581 } 2582 PRINT("ClientConnection::ProcessQueryEvent(): event: %p, type: %s:" 2583 " directory: (%" B_PRIdDEV ", %" B_PRIdINO ")\n", 2584 event, typeid(event).name(), volumeID, directoryID); 2585 2586 // create an array for the IDs of the client volumes a found entry may 2587 // reside on 2588 status_t result = B_OK; 2589 int32 volumeCount = fVolumes->Size(); 2590 int32* volumeIDs = new(std::nothrow) int32[volumeCount]; 2591 if (!volumeIDs) 2592 result = B_NO_MEMORY; 2593 ArrayDeleter<int32> volumeIDsDeleter(volumeIDs); 2594 2595 HasQueryPermissionClientVolumeFilter filter; 2596 2597 // load the directory the concerned entry belongs/belonged to 2598 Directory* directory; 2599 int32 concernedVolumes = 0; 2600 if (VolumeManager::GetDefault()->LoadDirectory(volumeID, directoryID, 2601 &directory) == B_OK) { 2602 // find out, which client volumes the directory is located in 2603 concernedVolumes = _GetContainingClientVolumes(directory, volumeIDs, 2604 volumeCount, &filter); 2605 } else { 2606 // Failed to load the directory, so maybe it has already been 2607 // deleted. For "entry removed" events, we consider all client 2608 // volumes to be notified -- those that don't know the entry will 2609 // ignore the event. 2610 if (event->opcode == B_ENTRY_REMOVED) { 2611 concernedVolumes = _GetAllClientVolumeIDs(volumeIDs, volumeCount, 2612 &filter); 2613 } 2614 } 2615 2616 // now push the event for each concerned client volume 2617 for (int32 i = 0; i < concernedVolumes; i++) 2618 _PushNodeMonitoringEvent(volumeIDs[i], event); 2619 // TODO: More than one volume will usually only be concerned in case of 2620 // nested client volumes. We could optimize the case by having an array of 2621 // volume IDs in the respective requests sent over the net (just as in the 2622 // ReadQueryReply). 2623 } 2624 2625 2626 // #pragma mark - 2627 2628 // _Close 2629 void 2630 ClientConnection::_Close() 2631 { 2632 // terminate node monitoring processor 2633 CloseNodeMonitoringEventQueue(); 2634 if (fNodeMonitoringProcessor >= 0 2635 && find_thread(NULL) != fNodeMonitoringProcessor) { 2636 int32 result; 2637 wait_for_thread(fNodeMonitoringProcessor, &result); 2638 // The variable is not unset, when this is the node monitoring 2639 // processor thread -- which is good, since the destructor will 2640 // wait for the thread in this case. 2641 fNodeMonitoringProcessor = -1; 2642 } 2643 if (fConnection) 2644 fConnection->Close(); 2645 // notify the listener 2646 ClientConnectionListener* listener = fListener; 2647 fListener = NULL; 2648 if (listener) 2649 listener->ClientConnectionClosed(this, fError); 2650 } 2651 2652 // _MarkClosed 2653 void 2654 ClientConnection::_MarkClosed(bool error) 2655 { 2656 AutoLocker<Locker> _(fLock); 2657 if (!fClosed) { 2658 fClosed = true; 2659 fError = error; 2660 } 2661 } 2662 2663 // _GetNodeInfo 2664 void 2665 ClientConnection::_GetNodeInfo(Node* node, NodeInfo* info) 2666 { 2667 if (node && info) { 2668 info->st = node->GetStat(); 2669 info->revision = VolumeManager::GetDefault()->GetRevision(); 2670 } 2671 } 2672 2673 // _GetEntryInfo 2674 void 2675 ClientConnection::_GetEntryInfo(Entry* entry, EntryInfo* info) 2676 { 2677 if (entry && info) { 2678 info->directoryID.volumeID = entry->GetVolumeID(); 2679 info->directoryID.nodeID = entry->GetDirectoryID(); 2680 info->name.SetTo(entry->GetName()); 2681 _GetNodeInfo(entry->GetNode(), &info->nodeInfo); 2682 } 2683 } 2684 2685 // _GetAttrInfo 2686 status_t 2687 ClientConnection::_GetAttrInfo(Request* request, const char* name, 2688 const attr_info& attrInfo, const void* data, AttributeInfo* info) 2689 { 2690 if (!request || !name || !info) 2691 return B_BAD_VALUE; 2692 2693 info->name.SetTo(name); 2694 info->info = attrInfo; 2695 data = (attrInfo.size > 0 ? data : NULL); 2696 int32 dataSize = (data ? attrInfo.size : 0); 2697 info->data.SetTo(data, dataSize); 2698 2699 // if the client has inverse endianess, swap the type, if we don't know it 2700 if (fInverseClientEndianess) { 2701 if (_KnownAttributeType(info->info.type)) { 2702 // we need to convert the data, if supplied 2703 if (data) { 2704 // allocate a buffer 2705 RequestBuffer* requestBuffer = RequestBuffer::Create(dataSize); 2706 if (!requestBuffer) 2707 return B_NO_MEMORY; 2708 2709 // convert the data 2710 memcpy(requestBuffer->GetData(), data, dataSize); 2711 _ConvertAttribute(info->info, requestBuffer->GetData()); 2712 } 2713 } else 2714 info->info.type = B_SWAP_INT32(info->info.type); 2715 } 2716 2717 return B_OK; 2718 } 2719 2720 // _GetAttrDirInfo 2721 status_t 2722 ClientConnection::_GetAttrDirInfo(Request* request, AttributeDirectory* attrDir, 2723 AttrDirInfo* info) 2724 { 2725 if (!request || !attrDir || !info || !attrDir->IsAttrDirValid()) 2726 return B_BAD_VALUE; 2727 2728 // add the attribute infos 2729 for (Attribute* attribute = attrDir->GetFirstAttribute(); 2730 attribute; 2731 attribute = attrDir->GetNextAttribute(attribute)) { 2732 // get the attribute info 2733 AttributeInfo attrInfo; 2734 attr_info bAttrInfo; 2735 attribute->GetInfo(&bAttrInfo); 2736 status_t error = _GetAttrInfo(request, attribute->GetName(), bAttrInfo, 2737 attribute->GetData(), &attrInfo); 2738 2739 // append it 2740 if (error == B_OK) 2741 error = info->attributeInfos.Append(attrInfo); 2742 if (error != B_OK) 2743 return error; 2744 } 2745 2746 info->revision = VolumeManager::GetDefault()->GetRevision(); 2747 info->isValid = true; 2748 2749 return B_OK; 2750 } 2751 2752 // _CreateVolume 2753 status_t 2754 ClientConnection::_CreateVolume(ClientVolume** _volume) 2755 { 2756 // create and init the volume 2757 ClientVolume* volume = new(std::nothrow) ClientVolume(fSecurityContextLock, 2758 this); 2759 if (!volume) 2760 return B_NO_MEMORY; 2761 status_t error = volume->Init(); 2762 if (error != B_OK) { 2763 delete volume; 2764 return error; 2765 } 2766 2767 // add it to the volume map 2768 AutoLocker<VolumeMap> locker(fVolumes); 2769 error = fVolumes->Put(volume->GetID(), volume); 2770 locker.Unlock(); 2771 2772 if (error == B_OK) 2773 *_volume = volume; 2774 else 2775 delete volume; 2776 2777 return error; 2778 } 2779 2780 // _GetVolume 2781 ClientVolume* 2782 ClientConnection::_GetVolume(int32 id) 2783 { 2784 AutoLocker<VolumeMap> _(fVolumes); 2785 ClientVolume* volume = fVolumes->Get(id); 2786 if (!volume || volume->IsRemoved()) 2787 return NULL; 2788 volume->AcquireReference(); 2789 return volume; 2790 } 2791 2792 // _PutVolume 2793 // 2794 // The VolumeManager may be locked, but no other lock must be held. 2795 void 2796 ClientConnection::_PutVolume(ClientVolume* volume) 2797 { 2798 if (!volume) 2799 return; 2800 2801 // decrement reference counter and remove the volume, if 0 2802 AutoLocker<VolumeMap> locker(fVolumes); 2803 bool removed = (volume->ReleaseReference() == 1 && volume->IsRemoved()); 2804 if (removed) 2805 fVolumes->Remove(volume->GetID()); 2806 locker.Unlock(); 2807 2808 if (removed) { 2809 VolumeManagerLocker managerLocker; 2810 delete volume; 2811 } 2812 } 2813 2814 // _UnmountVolume 2815 // 2816 // The caller must have a reference to the volume. 2817 void 2818 ClientConnection::_UnmountVolume(ClientVolume* volume) 2819 { 2820 if (!volume) 2821 return; 2822 AutoLocker<VolumeMap> locker(fVolumes); 2823 volume->MarkRemoved(); 2824 locker.Unlock(); 2825 2826 // push a notification event 2827 if (VolumeUnmountedEvent* event = new(std::nothrow) VolumeUnmountedEvent) { 2828 VolumeManagerLocker managerLocker; 2829 2830 event->opcode = B_DEVICE_UNMOUNTED; 2831 _PushNodeMonitoringEvent(volume->GetID(), event); 2832 event->ReleaseReference(); 2833 } 2834 } 2835 2836 // _UnmountAllVolumes 2837 void 2838 ClientConnection::_UnmountAllVolumes() 2839 { 2840 while (true) { 2841 // To avoid heap allocation (which can fail) we unmount the volumes 2842 // chunkwise. 2843 // get the volumes 2844 const int32 volumeChunkSize = 32; 2845 ClientVolume* volumes[volumeChunkSize]; 2846 int32 volumeCount = 0; 2847 AutoLocker<VolumeMap> volumesLocker(fVolumes); 2848 for (VolumeMap::Iterator it = fVolumes->GetIterator(); it.HasNext();) { 2849 if (ClientVolume* volume = _GetVolume(it.Next().value->GetID())) { 2850 volumes[volumeCount++] = volume; 2851 } 2852 if (volumeCount == volumeChunkSize) 2853 break; 2854 } 2855 volumesLocker.Unlock(); 2856 2857 // unmount and put the volumes 2858 for (int32 i = 0; i < volumeCount; i++) { 2859 ClientVolume* volume = volumes[i]; 2860 _UnmountVolume(volume); 2861 _PutVolume(volume); 2862 } 2863 2864 if (volumeCount < volumeChunkSize) 2865 break; 2866 } 2867 } 2868 2869 // _NodeMonitoringProcessorEntry 2870 int32 2871 ClientConnection::_NodeMonitoringProcessorEntry(void* data) 2872 { 2873 return ((ClientConnection*)data)->_NodeMonitoringProcessor(); 2874 } 2875 2876 2877 // _NodeMonitoringProcessor 2878 int32 2879 ClientConnection::_NodeMonitoringProcessor() 2880 { 2881 while (!fClosed) { 2882 // get the next request 2883 NodeMonitoringRequest* request = NULL; 2884 status_t error = fNodeMonitoringEvents->Pop(&request); 2885 2886 // get a client connection reference 2887 ConnectionReference connectionReference(this); 2888 if (!connectionReference.IsValid()) 2889 return B_OK; 2890 2891 // No request? Next round... 2892 if (error != B_OK) 2893 continue; 2894 ObjectDeleter<NodeMonitoringRequest> requestDeleter(request); 2895 2896 // send the request 2897 error = fConnection->SendRequest(request); 2898 if (error != B_OK) { 2899 ERROR(("ClientConnection::_NodeMonitoringProcessor(): " 2900 "Failed to send request.\n")); 2901 } 2902 } 2903 return 0; 2904 } 2905 2906 2907 // _PushNodeMonitoringEvent 2908 // 2909 // The caller must have a connection reference. Moreover the VolumeManager 2910 // must be locked. 2911 status_t 2912 ClientConnection::_PushNodeMonitoringEvent(int32 volumeID, 2913 NodeMonitoringEvent* event) 2914 { 2915 if (!event) 2916 return B_BAD_VALUE; 2917 2918 // get the volume 2919 ClientVolume* volume = _GetVolume(volumeID); 2920 if (!volume && event->opcode != B_DEVICE_UNMOUNTED) 2921 return B_BAD_VALUE; 2922 ClientVolumePutter volumePutter(this, volume); 2923 2924 // create a node monitoring request 2925 NodeMonitoringRequest* request = NULL; 2926 status_t error = B_ERROR; 2927 switch (event->opcode) { 2928 case B_ENTRY_CREATED: 2929 error = _EntryCreated(volume, 2930 dynamic_cast<EntryCreatedEvent*>(event), request); 2931 break; 2932 case B_ENTRY_REMOVED: 2933 error = _EntryRemoved(volume, 2934 dynamic_cast<EntryRemovedEvent*>(event), request); 2935 break; 2936 case B_ENTRY_MOVED: 2937 error = _EntryMoved(volume, 2938 dynamic_cast<EntryMovedEvent*>(event), request); 2939 break; 2940 case B_STAT_CHANGED: 2941 error = _NodeStatChanged(volume, 2942 dynamic_cast<StatChangedEvent*>(event), request); 2943 break; 2944 case B_ATTR_CHANGED: 2945 error = _NodeAttributeChanged(volume, 2946 dynamic_cast<AttributeChangedEvent*>(event), request); 2947 break; 2948 case B_DEVICE_UNMOUNTED: 2949 error = B_OK; 2950 break; 2951 } 2952 2953 // replace all data buffers -- when the request is actually sent, they 2954 // might no longer exist 2955 if (error == B_OK) 2956 error = RequestBufferReplacer().ReplaceBuffer(request); 2957 2958 if (error == B_OK) { 2959 // common initialization 2960 request->volumeID = volumeID; 2961 request->opcode = event->opcode; 2962 request->revision = VolumeManager::GetDefault()->GetRevision(); 2963 2964 // push the request 2965 error = fNodeMonitoringEvents->Push(request); 2966 if (error != B_OK) 2967 delete request; 2968 } 2969 2970 return error; 2971 } 2972 2973 // _EntryCreated 2974 status_t 2975 ClientConnection::_EntryCreated(ClientVolume* volume, EntryCreatedEvent* event, 2976 NodeMonitoringRequest*& _request) 2977 { 2978 // allocate the request 2979 EntryCreatedRequest* request = new(std::nothrow) EntryCreatedRequest; 2980 if (!request) 2981 return B_NO_MEMORY; 2982 ObjectDeleter<NodeMonitoringRequest> requestDeleter(request); 2983 2984 // get the name 2985 const char* name = event->name.GetString(); 2986 2987 // set the request fields 2988 request->directoryID = NodeID(event->volumeID, event->directoryID); 2989 request->nodeID = NodeID(event->volumeID, event->nodeID); 2990 request->name.SetTo(name); 2991 if (event->queryHandler) { 2992 request->port = event->remotePort; 2993 request->token = event->remoteToken; 2994 request->queryUpdate = true; 2995 } else 2996 request->queryUpdate = false; 2997 2998 // try to get an entry info 2999 Entry* entry; 3000 if (VolumeManager::GetDefault()->LoadEntry(event->volumeID, 3001 event->directoryID, name, true, &entry) == B_OK 3002 && entry->GetNode()->GetVolumeID() == event->volumeID 3003 && entry->GetNode()->GetID() == event->nodeID) { 3004 _GetEntryInfo(entry, &request->entryInfo); 3005 request->entryInfoValid = true; 3006 } else 3007 request->entryInfoValid = false; 3008 3009 requestDeleter.Detach(); 3010 _request = request; 3011 return B_OK; 3012 } 3013 3014 // _EntryRemoved 3015 status_t 3016 ClientConnection::_EntryRemoved(ClientVolume* volume, EntryRemovedEvent* event, 3017 NodeMonitoringRequest*& _request) 3018 { 3019 // special handling, if it is the root node of the client volume that has 3020 // been removed 3021 if (!event->queryHandler 3022 && NodeRef(event->nodeVolumeID, event->nodeID) 3023 == volume->GetRootNodeRef()) { 3024 NoAllocEntryRef ref(event->nodeVolumeID, event->nodeID, "."); 3025 BEntry entry; 3026 if (FDManager::SetEntry(&entry, &ref) != B_OK || !entry.Exists()) 3027 _UnmountVolume(volume); 3028 3029 // don't send the "entry removed" event 3030 return B_ERROR; 3031 } 3032 3033 // allocate the request 3034 EntryRemovedRequest* request = new(std::nothrow) EntryRemovedRequest; 3035 if (!request) 3036 return B_NO_MEMORY; 3037 ObjectDeleter<NodeMonitoringRequest> requestDeleter(request); 3038 3039 // get the name 3040 const char* name = event->name.GetString(); 3041 3042 // set the request fields 3043 request->directoryID = NodeID(event->volumeID, event->directoryID); 3044 request->nodeID = NodeID(event->nodeVolumeID, event->nodeID); 3045 request->name.SetTo(name); 3046 if (event->queryHandler) { 3047 request->port = event->remotePort; 3048 request->token = event->remoteToken; 3049 request->queryUpdate = true; 3050 } else 3051 request->queryUpdate = false; 3052 3053 requestDeleter.Detach(); 3054 _request = request; 3055 return B_OK; 3056 } 3057 3058 // _EntryMoved 3059 status_t 3060 ClientConnection::_EntryMoved(ClientVolume* volume, EntryMovedEvent* event, 3061 NodeMonitoringRequest*& _request) 3062 { 3063 // allocate the request 3064 EntryMovedRequest* request = new(std::nothrow) EntryMovedRequest; 3065 if (!request) 3066 return B_NO_MEMORY; 3067 ObjectDeleter<NodeMonitoringRequest> requestDeleter(request); 3068 3069 // allocate memory for the names 3070 int32 fromNameLen = event->fromName.GetLength(); 3071 const char* fromName 3072 = (fromNameLen > 0 ? event->fromName.GetString() : NULL); 3073 const char* toName = event->toName.GetString(); 3074 3075 // set the request fields 3076 request->fromDirectoryID = NodeID(event->volumeID, event->fromDirectoryID); 3077 request->toDirectoryID = NodeID(event->volumeID, event->toDirectoryID); 3078 request->nodeID = NodeID(event->nodeVolumeID, event->nodeID); 3079 request->fromName.SetTo(fromName); 3080 request->toName.SetTo(toName); 3081 request->queryUpdate = false; 3082 3083 // try to get an entry info 3084 Entry* entry; 3085 if (VolumeManager::GetDefault()->LoadEntry(event->volumeID, 3086 event->toDirectoryID, toName, true, &entry) == B_OK 3087 && entry->GetNode()->GetVolumeID() == event->nodeVolumeID 3088 && entry->GetNode()->GetID() == event->nodeID) { 3089 _GetEntryInfo(entry, &request->entryInfo); 3090 request->entryInfoValid = true; 3091 } else 3092 request->entryInfoValid = false; 3093 3094 requestDeleter.Detach(); 3095 _request = request; 3096 return B_OK; 3097 } 3098 3099 // _NodeStatChanged 3100 status_t 3101 ClientConnection::_NodeStatChanged(ClientVolume* volume, 3102 StatChangedEvent* event, NodeMonitoringRequest*& _request) 3103 { 3104 // get the node 3105 Node* node = volume->GetNode(event->volumeID, event->nodeID); 3106 if (!node) 3107 return B_ENTRY_NOT_FOUND; 3108 3109 // allocate the request 3110 StatChangedRequest* request = new(std::nothrow) StatChangedRequest; 3111 if (!request) 3112 return B_NO_MEMORY; 3113 ObjectDeleter<NodeMonitoringRequest> requestDeleter(request); 3114 3115 // set the request fields 3116 request->nodeID = NodeID(event->volumeID, event->nodeID); 3117 _GetNodeInfo(node, &request->nodeInfo); 3118 request->queryUpdate = false; 3119 3120 requestDeleter.Detach(); 3121 _request = request; 3122 return B_OK; 3123 } 3124 3125 // _NodeAttributeChanged 3126 status_t 3127 ClientConnection::_NodeAttributeChanged(ClientVolume* volume, 3128 AttributeChangedEvent* event, NodeMonitoringRequest*& _request) 3129 { 3130 // get the node 3131 Node* node = volume->GetNode(event->volumeID, event->nodeID); 3132 if (!node) 3133 return B_ENTRY_NOT_FOUND; 3134 3135 // update the attribute directory 3136 bool removed = false; 3137 bool valid = false; 3138 attr_info info; 3139 const void* data = NULL; 3140 status_t error = node->UpdateAttribute(event->attribute.GetString(), 3141 &removed, &info, &data); 3142 valid = (error == B_OK); 3143 3144 // allocate the request 3145 AttributeChangedRequest* request = new(std::nothrow) AttributeChangedRequest; 3146 if (!request) 3147 return B_NO_MEMORY; 3148 ObjectDeleter<NodeMonitoringRequest> requestDeleter(request); 3149 3150 // get an attr dir info, if the directory is valid 3151 if (node->IsAttrDirValid()) { 3152 status_t error = _GetAttrDirInfo(request, node, &request->attrDirInfo); 3153 if (error != B_OK) 3154 return error; 3155 } 3156 3157 // get name and the data size 3158 int32 dataSize = (data ? info.size : 0); 3159 const char* name = event->attribute.GetString(); 3160 3161 // set the request fields 3162 request->nodeID = NodeID(event->volumeID, event->nodeID); 3163 request->attrInfo.name.SetTo(name); 3164 request->valid = valid; 3165 request->removed = removed; 3166 if (!removed && valid) { 3167 request->attrInfo.info = info; 3168 request->attrInfo.data.SetTo(data, dataSize); 3169 } 3170 request->queryUpdate = false; 3171 3172 requestDeleter.Detach(); 3173 _request = request; 3174 return B_OK; 3175 } 3176 3177 // _KnownAttributeType 3178 bool 3179 ClientConnection::_KnownAttributeType(type_code type) 3180 { 3181 if (!fInverseClientEndianess) 3182 return false; 3183 3184 switch (type) { 3185 case B_BOOL_TYPE: 3186 case B_CHAR_TYPE: 3187 case B_COLOR_8_BIT_TYPE: 3188 case B_DOUBLE_TYPE: 3189 case B_FLOAT_TYPE: 3190 case B_GRAYSCALE_8_BIT_TYPE: 3191 case B_INT64_TYPE: 3192 case B_INT32_TYPE: 3193 case B_INT16_TYPE: 3194 case B_INT8_TYPE: 3195 case B_MESSAGE_TYPE: 3196 case B_MESSENGER_TYPE: 3197 case B_MIME_TYPE: 3198 case B_MONOCHROME_1_BIT_TYPE: 3199 case B_OFF_T_TYPE: 3200 case B_POINTER_TYPE: 3201 case B_POINT_TYPE: 3202 case B_RECT_TYPE: 3203 case B_REF_TYPE: 3204 case B_RGB_COLOR_TYPE: 3205 case B_SIZE_T_TYPE: 3206 case B_SSIZE_T_TYPE: 3207 case B_STRING_TYPE: 3208 case B_TIME_TYPE: 3209 case B_UINT64_TYPE: 3210 case B_UINT32_TYPE: 3211 case B_UINT16_TYPE: 3212 case B_UINT8_TYPE: 3213 case B_ASCII_TYPE: 3214 case B_MIME_STRING_TYPE: 3215 return true; 3216 3217 //B_RGB_32_BIT_TYPE: We could translate it, but it's heavy... 3218 } 3219 3220 return false; 3221 } 3222 3223 // _ConvertAttribute 3224 void 3225 ClientConnection::_ConvertAttribute(const attr_info& info, void* buffer) 3226 { 3227 swap_data(info.type, buffer, info.size, B_SWAP_ALWAYS); 3228 } 3229 3230 3231 // #pragma mark - 3232 3233 // _OpenQuery 3234 status_t 3235 ClientConnection::_OpenQuery(const char* queryString, uint32 flags, 3236 port_id remotePort, int32 remoteToken, QueryHandle** _handle) 3237 { 3238 if (!queryString || !_handle) 3239 return B_BAD_VALUE; 3240 3241 // open query 3242 QueryHandle* queryHandle; 3243 status_t error = VolumeManager::GetDefault()->OpenQuery(this, queryString, 3244 flags, remotePort, remoteToken, &queryHandle); 3245 if (error != B_OK) 3246 return error; 3247 BReference<QueryHandle> handleReference(queryHandle, true); 3248 3249 // lock the handle 3250 queryHandle->Lock(); 3251 3252 // add the handle 3253 error = fQueryHandles->AddNodeHandle(queryHandle); 3254 if (error != B_OK) 3255 return error; 3256 3257 handleReference.Detach(); 3258 *_handle = queryHandle; 3259 return B_OK; 3260 } 3261 3262 // _CloseQuery 3263 status_t 3264 ClientConnection::_CloseQuery(QueryHandle* handle) 3265 { 3266 if (!handle || !fQueryHandles->RemoveNodeHandle(handle)) 3267 return B_BAD_VALUE; 3268 3269 return B_OK; 3270 } 3271 3272 // _LockQueryHandle 3273 // 3274 // VolumeManager must NOT be locked. 3275 status_t 3276 ClientConnection::_LockQueryHandle(int32 cookie, QueryHandle** _handle) 3277 { 3278 NodeHandle* handle; 3279 status_t error = fQueryHandles->LockNodeHandle(cookie, &handle); 3280 if (error == B_OK) 3281 *_handle = static_cast<QueryHandle*>(handle); 3282 return error; 3283 } 3284 3285 // _UnlockQueryHandle 3286 // 3287 // VolumeManager may or may not be locked. 3288 void 3289 ClientConnection::_UnlockQueryHandle(NodeHandle* nodeHandle) 3290 { 3291 fQueryHandles->UnlockNodeHandle(nodeHandle); 3292 } 3293 3294 3295 // #pragma mark - 3296 3297 // _GetAllClientVolumeIDs 3298 int32 3299 ClientConnection::_GetAllClientVolumeIDs(int32* volumeIDs, int32 arraySize, 3300 ClientVolumeFilter* filter) 3301 { 3302 int32 count = 0; 3303 AutoLocker<VolumeMap> volumesLocker(fVolumes); 3304 for (VolumeMap::Iterator it = fVolumes->GetIterator(); 3305 it.HasNext() && arraySize > count;) { 3306 ClientVolume* clientVolume = it.Next().value; 3307 if (!filter || filter->FilterVolume(this, clientVolume)) 3308 volumeIDs[count++] = clientVolume->GetID(); 3309 } 3310 3311 return count; 3312 } 3313 3314 // _GetContainingClientVolumes 3315 int32 3316 ClientConnection::_GetContainingClientVolumes(Directory* directory, 3317 int32* volumeIDs, int32 arraySize, ClientVolumeFilter* filter) 3318 { 3319 int32 count = 0; 3320 VolumeManager* volumeManager = VolumeManager::GetDefault(); 3321 AutoLocker<VolumeMap> volumesLocker(fVolumes); 3322 for (VolumeMap::Iterator it = fVolumes->GetIterator(); 3323 it.HasNext() && arraySize > count;) { 3324 ClientVolume* clientVolume = it.Next().value; 3325 Directory* clientVolumeRoot = clientVolume->GetRootDirectory(); 3326 if (volumeManager->DirectoryContains(clientVolumeRoot, directory, true) 3327 && (!filter || filter->FilterVolume(this, clientVolume))) { 3328 volumeIDs[count++] = clientVolume->GetID(); 3329 } 3330 } 3331 3332 return count; 3333 } 3334 3335 3336 // #pragma mark - 3337 // #pragma mark ----- ClientConnectionListener ----- 3338 3339 // constructor 3340 ClientConnectionListener::ClientConnectionListener() 3341 { 3342 } 3343 3344 // destructor 3345 ClientConnectionListener::~ClientConnectionListener() 3346 { 3347 } 3348 3349