xref: /haiku/src/servers/registrar/MessagingService.cpp (revision d3d8b26997fac34a84981e6d2b649521de2cc45a)
1 /*
2  * Copyright 2005, Ingo Weinhold, bonefish@users.sf.net. All rights reserved.
3  * Distributed under the terms of the MIT License.
4  */
5 
6 #include <map>
7 #include <new>
8 
9 #include <string.h>
10 
11 #include <Autolock.h>
12 
13 #include <syscalls.h>
14 
15 #include "Debug.h"
16 #include "MessageDeliverer.h"
17 #include "MessagingService.h"
18 
19 using std::map;
20 using std::nothrow;
21 
22 // sService -- the singleton instance
23 MessagingService *MessagingService::sService = NULL;
24 
25 /*!	\class MessagingArea
26 	\brief Represents an area of the messaging service shared between kernel
27 		   and registrar.
28 
29 	The main purpose of the class is to retrieve (and remove) commands from
30 	the area.
31 */
32 
33 // constructor
34 MessagingArea::MessagingArea()
35 {
36 }
37 
38 // destructor
39 MessagingArea::~MessagingArea()
40 {
41 	if (fID >= 0)
42 		delete_area(fID);
43 }
44 
45 // Create
46 status_t
47 MessagingArea::Create(area_id kernelAreaID, sem_id lockSem, sem_id counterSem,
48 	MessagingArea *&_area)
49 {
50 	// allocate the object on the heap
51 	MessagingArea *area = new(nothrow) MessagingArea;
52 	if (!area)
53 		return B_NO_MEMORY;
54 
55 	// clone the kernel area
56 	area_id areaID = clone_area("messaging", (void**)&area->fHeader,
57 		B_ANY_ADDRESS, B_READ_AREA | B_WRITE_AREA, kernelAreaID);
58 	if (areaID < 0) {
59 		delete area;
60 		return areaID;
61 	}
62 
63 	// finish the initialization of the object
64 	area->fID = areaID;
65 	area->fSize = area->fHeader->size;
66 	area->fLockSem = lockSem;
67 	area->fCounterSem = counterSem;
68 	area->fNextArea = NULL;
69 
70 	_area = area;
71 	return B_OK;
72 }
73 
74 // Lock
75 bool
76 MessagingArea::Lock()
77 {
78 	// benaphore-like locking
79 	if (atomic_add(&fHeader->lock_counter, 1) == 0)
80 		return true;
81 
82 	return (acquire_sem(fLockSem) == B_OK);
83 }
84 
85 // Unlock
86 void
87 MessagingArea::Unlock()
88 {
89 	if (atomic_add(&fHeader->lock_counter, -1) > 1)
90 		release_sem(fLockSem);
91 }
92 
93 // ID
94 area_id
95 MessagingArea::ID() const
96 {
97 	return fID;
98 }
99 
100 // Size
101 int32
102 MessagingArea::Size() const
103 {
104 	return fSize;
105 }
106 
107 // CountCommands
108 int32
109 MessagingArea::CountCommands() const
110 {
111 	return fHeader->command_count;
112 }
113 
114 // PopCommand
115 const messaging_command *
116 MessagingArea::PopCommand()
117 {
118 	if (fHeader->command_count == 0)
119 		return NULL;
120 
121 	// get the command
122 	messaging_command *command
123 		= (messaging_command*)((char*)fHeader + fHeader->first_command);
124 
125 	// remove it from the area
126 	// (as long as the area is still locked, noone will overwrite the contents)
127 	if (--fHeader->command_count == 0)
128 		fHeader->first_command = fHeader->last_command = 0;
129 	else
130 		fHeader->first_command = command->next_command;
131 
132 	return command;
133 }
134 
135 // Discard
136 void
137 MessagingArea::Discard()
138 {
139 	fHeader->size = 0;
140 }
141 
142 // NextKernelAreaID
143 area_id
144 MessagingArea::NextKernelAreaID() const
145 {
146 	return fHeader->next_kernel_area;
147 }
148 
149 // SetNextArea
150 void
151 MessagingArea::SetNextArea(MessagingArea *area)
152 {
153 	fNextArea = area;
154 }
155 
156 // NextArea
157 MessagingArea *
158 MessagingArea::NextArea() const
159 {
160 	return fNextArea;
161 }
162 
163 
164 // #pragma mark -
165 
166 // constructor
167 MessagingCommandHandler::MessagingCommandHandler()
168 {
169 }
170 
171 // destructor
172 MessagingCommandHandler::~MessagingCommandHandler()
173 {
174 }
175 
176 
177 // #pragma mark -
178 
179 // DefaultSendCommandHandler
180 class MessagingService::DefaultSendCommandHandler
181 	: public MessagingCommandHandler {
182 
183 	virtual void HandleMessagingCommand(uint32 _command, const void *data,
184 		int32 dataSize)
185 	{
186 		const messaging_command_send_message *sendData
187 			= (const messaging_command_send_message*)data;
188 		const void *messageData = (uint8*)data
189 			+ sizeof(messaging_command_send_message)
190 			+ sizeof(messaging_target) * sendData->target_count;
191 
192 		DefaultMessagingTargetSet set(sendData->targets,
193 			sendData->target_count);
194 		MessageDeliverer::Default()->DeliverMessage(messageData,
195 			sendData->message_size, set);
196 	}
197 };
198 
199 // CommandHandlerMap
200 struct MessagingService::CommandHandlerMap
201 	: map<uint32, MessagingCommandHandler*> {
202 };
203 
204 
205 /*! \class MessagingService
206 	\brief Userland implementation of the kernel -> userland messaging service.
207 
208 	This service provides a way for the kernel to send BMessages (usually
209 	notification (e.g. node monitoring) messages) to userland applications.
210 
211 	The kernel could write the messages directly to the respective target ports,
212 	but this has the disadvantage, that a message needs to be dropped, if the
213 	port is full at the moment of sending. By transferring the message to the
214 	registrar, it is possible to use the MessageDeliverer which retries sending
215 	messages on full ports.
216 
217 	The message transfer is implemented via areas shared between kernel
218 	and registrar. By default one area is used as a ring buffer. The kernel
219 	adds messages to it, the registrar removes them. If the area is full, the
220 	kernel creates a new one and adds it to the area list.
221 
222 	While the service is called `messaging service' and we were speaking of
223 	`messages' being passed through the areas, the service is actually more
224 	general. In fact `commands' are passed through the areas. Currently the
225 	only implemented command type is to send a message, but it is very easy
226 	to add further command types (e.g. one for alerting the user in case of
227 	errors).
228 
229 	The MessagingService maintains a mapping of command types to command
230 	handlers (MessagingCommandHandler, which perform the actual processing
231 	of the commands), that can be altered via
232 	MessagingService::SetCommandHandler().
233 */
234 
235 // constructor
236 MessagingService::MessagingService()
237 	: fLock("messaging service"),
238 	  fLockSem(-1),
239 	  fCounterSem(-1),
240 	  fFirstArea(NULL),
241 	  fCommandHandlers(NULL),
242 	  fCommandProcessor(-1),
243 	  fTerminating(false)
244 {
245 }
246 
247 // destructor
248 MessagingService::~MessagingService()
249 {
250 	fTerminating = true;
251 
252 	if (fLockSem >= 0)
253 		delete_sem(fLockSem);
254 	if (fCounterSem >= 0)
255 		delete_sem(fCounterSem);
256 
257 	if (fCommandProcessor >= 0) {
258 		int32 result;
259 		wait_for_thread(fCommandProcessor, &result);
260 	}
261 
262 	delete fCommandHandlers;
263 
264 	delete fFirstArea;
265 }
266 
267 // Init
268 status_t
269 MessagingService::Init()
270 {
271 	// create the semaphores
272 	fLockSem = create_sem(0, "messaging lock");
273 	if (fLockSem < 0)
274 		return fLockSem;
275 
276 	fCounterSem = create_sem(0, "messaging counter");
277 	if (fCounterSem < 0)
278 		return fCounterSem;
279 
280 	// create the command handler map
281 	fCommandHandlers = new(nothrow) CommandHandlerMap;
282 	if (!fCommandHandlers)
283 		return B_NO_MEMORY;
284 
285 	// spawn the command processor
286 	fCommandProcessor = spawn_thread(MessagingService::_CommandProcessorEntry,
287 		"messaging command processor", B_DISPLAY_PRIORITY, this);
288 	if (fCommandProcessor < 0)
289 		return fCommandProcessor;
290 
291 	// register with the kernel
292 	area_id areaID = _kern_register_messaging_service(fLockSem, fCounterSem);
293 	if (areaID < 0)
294 		return areaID;
295 
296 	// create the area
297 	status_t error = MessagingArea::Create(areaID, fLockSem, fCounterSem,
298 		fFirstArea);
299 	if (error != B_OK) {
300 		_kern_unregister_messaging_service();
301 		return error;
302 	}
303 
304 	// resume the command processor
305 	resume_thread(fCommandProcessor);
306 
307 	// install the default send message command handler
308 	MessagingCommandHandler *handler = new(nothrow) DefaultSendCommandHandler;
309 	if (!handler)
310 		return B_NO_MEMORY;
311 	SetCommandHandler(MESSAGING_COMMAND_SEND_MESSAGE, handler);
312 
313 	return B_OK;
314 }
315 
316 // CreateDefault
317 status_t
318 MessagingService::CreateDefault()
319 {
320 	if (sService)
321 		return B_OK;
322 
323 	// create the service
324 	MessagingService *service = new(nothrow) MessagingService;
325 	if (!service)
326 		return B_NO_MEMORY;
327 
328 	// init it
329 	status_t error = service->Init();
330 	if (error != B_OK) {
331 		delete service;
332 		return error;
333 	}
334 
335 	sService = service;
336 	return B_OK;
337 }
338 
339 // DeleteDefault
340 void
341 MessagingService::DeleteDefault()
342 {
343 	if (sService) {
344 		delete sService;
345 		sService = NULL;
346 	}
347 }
348 
349 // Default
350 MessagingService *
351 MessagingService::Default()
352 {
353 	return sService;
354 }
355 
356 // SetCommandHandler
357 void
358 MessagingService::SetCommandHandler(uint32 command,
359 	MessagingCommandHandler *handler)
360 {
361 	BAutolock _(fLock);
362 
363 	if (handler) {
364 		(*fCommandHandlers)[command] = handler;
365 	} else {
366 		// no handler: remove and existing entry
367 		CommandHandlerMap::iterator it = fCommandHandlers->find(command);
368 		if (it != fCommandHandlers->end())
369 			fCommandHandlers->erase(it);
370 	}
371 }
372 
373 // _GetCommandHandler
374 MessagingCommandHandler *
375 MessagingService::_GetCommandHandler(uint32 command) const
376 {
377 	BAutolock _(fLock);
378 
379 	CommandHandlerMap::iterator it = fCommandHandlers->find(command);
380 	return (it != fCommandHandlers->end() ? it->second : NULL);
381 }
382 
383 // _CommandProcessorEntry
384 int32
385 MessagingService::_CommandProcessorEntry(void *data)
386 {
387 	return ((MessagingService*)data)->_CommandProcessor();
388 }
389 
390 // _CommandProcessor
391 int32
392 MessagingService::_CommandProcessor()
393 {
394 	bool commandWaiting = false;
395 	while (!fTerminating) {
396 		// wait for the next command
397 		if (!commandWaiting) {
398 			status_t error = acquire_sem(fCounterSem);
399 			if (error != B_OK)
400 				continue;
401 		} else
402 			commandWaiting = false;
403 
404 		// get it from the first area
405 		MessagingArea *area = fFirstArea;
406 		area->Lock();
407 		while (area->CountCommands() > 0) {
408 			const messaging_command *command = area->PopCommand();
409 			if (!command) {
410 				// something's seriously wrong
411 				ERROR(("MessagingService::_CommandProcessor(): area %p (%ld) "
412 					"has command count %ld, but doesn't return any more "
413 					"commands.", area, area->ID(), area->CountCommands()));
414 				break;
415 			}
416 PRINT(("MessagingService::_CommandProcessor(): got command %lu\n",
417 command->command));
418 
419 			// dispatch the command
420 			MessagingCommandHandler *handler
421 				= _GetCommandHandler(command->command);
422 			if (handler) {
423 				handler->HandleMessagingCommand(command->command, command->data,
424 					command->size - sizeof(messaging_command));
425 			} else {
426 				WARNING(("MessagingService::_CommandProcessor(): No handler "
427 					"found for command %lu\n", command->command));
428 			}
429 		}
430 
431 		// there is a new area we don't know yet
432 		if (!area->NextArea() && area->NextKernelAreaID() >= 0) {
433 			// create it
434 			MessagingArea *nextArea;
435 			status_t error = MessagingArea::Create(area->NextKernelAreaID(),
436 				fLockSem, fCounterSem, nextArea);
437 			if (error == B_OK) {
438 				area->SetNextArea(nextArea);
439 				commandWaiting = true;
440 			} else {
441 				// Bad, but what can we do?
442 				ERROR(("MessagingService::_CommandProcessor(): Failed to clone "
443 					"kernel area %ld: %s\n", area->NextKernelAreaID(),
444 					strerror(error)));
445 			}
446 
447 		}
448 
449 		// if the current area is empty and there is a next one, we discard the
450 		// current one
451 		if (area->NextArea() && area->CountCommands() == 0) {
452 			fFirstArea = area->NextArea();
453 			area->Discard();
454 			area->Unlock();
455 			delete area;
456 		} else {
457 			area->Unlock();
458 		}
459 	}
460 
461 	return 0;
462 }
463 
464