xref: /haiku/src/add-ons/kernel/file_systems/netfs/server/ClientConnection.cpp (revision 4c8e85b316c35a9161f5a1c50ad70bc91c83a76f)
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