 0f7f3c0d54
			
		
	
	
		0f7f3c0d54
		
	
	
	
	
		
			
			- internal structure rearrangement; - respond to char device open requests to avoid hanging VFS threads; - make drivers use designated initializers; - use devminor_t for all minor device numbers; - change bdr_other hook to take ipc_status and return nothing; - fix default geometry computation; - add support for sef_cancel. Change-Id: Ia063a136a3ddb2b78de36180feda870605753d70
		
			
				
	
	
		
			95 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			95 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* This file contains the singlethreaded device driver interface.
 | |
|  *
 | |
|  * Changes:
 | |
|  *   Aug 27, 2011   extracted from driver.c (A. Welzel)
 | |
|  *
 | |
|  * The entry points into this file are:
 | |
|  *   blockdriver_task:		the main message loop of the driver
 | |
|  *   blockdriver_terminate:	break out of the main message loop
 | |
|  *   blockdriver_receive_mq:	message receive interface with message queueing
 | |
|  *   blockdriver_mq_queue:	queue an incoming message for later processing
 | |
|  */
 | |
| 
 | |
| #include <minix/drivers.h>
 | |
| #include <minix/blockdriver.h>
 | |
| 
 | |
| #include "const.h"
 | |
| #include "driver.h"
 | |
| #include "mq.h"
 | |
| 
 | |
| static int running;
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *			       blockdriver_receive_mq			     *
 | |
|  *===========================================================================*/
 | |
| int blockdriver_receive_mq(message *m_ptr, int *status_ptr)
 | |
| {
 | |
| /* receive() interface for drivers with message queueing. */
 | |
| 
 | |
|   /* Any queued messages? */
 | |
|   if (mq_dequeue(SINGLE_THREAD, m_ptr, status_ptr))
 | |
| 	return OK;
 | |
| 
 | |
|   /* Fall back to standard receive() interface otherwise. */
 | |
|   return driver_receive(ANY, m_ptr, status_ptr);
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				blockdriver_terminate			     *
 | |
|  *===========================================================================*/
 | |
| void blockdriver_terminate(void)
 | |
| {
 | |
| /* Break out of the main driver loop after finishing the current request. */
 | |
| 
 | |
|   running = FALSE;
 | |
| 
 | |
|   sef_cancel();
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				blockdriver_task			     *
 | |
|  *===========================================================================*/
 | |
| void blockdriver_task(struct blockdriver *bdp)
 | |
| {
 | |
| /* Main program of any block device driver task. */
 | |
|   int r, ipc_status;
 | |
|   message mess;
 | |
| 
 | |
|   running = TRUE;
 | |
| 
 | |
|   /* Here is the main loop of the disk task.  It waits for a message, carries
 | |
|    * it out, and sends a reply.
 | |
|    */
 | |
|   while (running) {
 | |
| 	if ((r = blockdriver_receive_mq(&mess, &ipc_status)) != OK) {
 | |
| 		if (r == EINTR && !running)
 | |
| 			break;
 | |
| 
 | |
| 		panic("blockdriver_receive_mq failed: %d", r);
 | |
| 	}
 | |
| 
 | |
| 	blockdriver_process(bdp, &mess, ipc_status);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				blockdriver_process			     *
 | |
|  *===========================================================================*/
 | |
| void blockdriver_process(struct blockdriver *bdp, message *m_ptr,
 | |
|   int ipc_status)
 | |
| {
 | |
| /* Handle the given received message. */
 | |
| 
 | |
|   blockdriver_process_on_thread(bdp, m_ptr, ipc_status, SINGLE_THREAD);
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				blockdriver_mq_queue			     *
 | |
|  *===========================================================================*/
 | |
| int blockdriver_mq_queue(message *m, int status)
 | |
| {
 | |
| /* Queue a message for later processing. */
 | |
| 
 | |
|   return mq_enqueue(SINGLE_THREAD, m, status);
 | |
| }
 |