 2bfeeed885
			
		
	
	
		2bfeeed885
		
	
	
	
	
		
			
			. all invocations were S or D, so can safely be dropped to prepare for the segmentless world . still assign D to the SCP_SEG field in the message to make previous kernels usable
		
			
				
	
	
		
			509 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			509 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* This file contains the printer driver. It is a fairly simple driver,
 | |
|  * supporting only one printer.  Characters that are written to the driver
 | |
|  * are written to the printer without any changes at all.
 | |
|  *
 | |
|  * Changes:
 | |
|  *	May 07, 2004	fix: wait until printer is ready  (Jorrit N. Herder)
 | |
|  *	May 06, 2004	printer driver moved to user-space  (Jorrit N. Herder) 
 | |
|  *
 | |
|  * The valid messages and their parameters are:
 | |
|  *
 | |
|  *   DEV_OPEN:	initializes the printer
 | |
|  *   DEV_CLOSE:	does nothing
 | |
|  *   HARD_INT:	interrupt handler has finished current chunk of output
 | |
|  *   DEV_WRITE:	a process wants to write on a terminal
 | |
|  *   CANCEL:	terminate a previous incomplete system call immediately
 | |
|  *
 | |
|  *    m_type      TTY_LINE  USER_ENDPT  COUNT    ADDRESS
 | |
|  * |-------------+---------+---------+---------+---------|
 | |
|  * | DEV_OPEN    |         |         |         |         |
 | |
|  * |-------------+---------+---------+---------+---------|
 | |
|  * | DEV_CLOSE   |         | proc nr |         |         |
 | |
|  * -------------------------------------------------------
 | |
|  * | HARD_INT    |         |         |         |         |
 | |
|  * |-------------+---------+---------+---------+---------|
 | |
|  * | SYS_EVENT   |         |         |         |         |
 | |
|  * |-------------+---------+---------+---------+---------|
 | |
|  * | DEV_WRITE   |minor dev| proc nr |  count  | buf ptr |
 | |
|  * |-------------+---------+---------+---------+---------|
 | |
|  * | CANCEL      |minor dev| proc nr |         |         |
 | |
|  * -------------------------------------------------------
 | |
|  * 
 | |
|  * Note: since only 1 printer is supported, minor dev is not used at present.
 | |
|  */
 | |
| 
 | |
| #include <minix/endpoint.h>
 | |
| #include <minix/drivers.h>
 | |
| #include <minix/chardriver.h>
 | |
| 
 | |
| /* Control bits (in port_base + 2).  "+" means positive logic and "-" means
 | |
|  * negative logic.  Most of the signals are negative logic on the pins but
 | |
|  * many are converted to positive logic in the ports.  Some manuals are
 | |
|  * misleading because they only document the pin logic.
 | |
|  *
 | |
|  *	+0x01	Pin 1	-Strobe
 | |
|  *	+0x02	Pin 14	-Auto Feed
 | |
|  *	-0x04	Pin 16	-Initialize Printer
 | |
|  *	+0x08	Pin 17	-Select Printer
 | |
|  *	+0x10	IRQ7 Enable
 | |
|  *
 | |
|  * Auto Feed and Select Printer are always enabled. Strobe is enabled briefly
 | |
|  * when characters are output.  Initialize Printer is enabled briefly when
 | |
|  * the task is started.  IRQ7 is enabled when the first character is output
 | |
|  * and left enabled until output is completed (or later after certain
 | |
|  * abnormal completions).
 | |
|  */
 | |
| #define ASSERT_STROBE   0x1D	/* strobe a character to the interface */
 | |
| #define NEGATE_STROBE   0x1C	/* enable interrupt on interface */
 | |
| #define PR_SELECT          0x0C	/* select printer bit */
 | |
| #define INIT_PRINTER    0x08	/* init printer bits */
 | |
| 
 | |
| /* Status bits (in port_base + 2).
 | |
|  *
 | |
|  *	-0x08	Pin 15	-Error
 | |
|  *	+0x10	Pin 13	+Select Status
 | |
|  *	+0x20	Pin 12	+Out of Paper
 | |
|  *	-0x40	Pin 10	-Acknowledge
 | |
|  *	-0x80	Pin 11	+Busy
 | |
|  */
 | |
| #define BUSY_STATUS     0x10	/* printer gives this status when busy */
 | |
| #define NO_PAPER        0x20	/* status bit saying that paper is out */
 | |
| #define NORMAL_STATUS   0x90	/* printer gives this status when idle */
 | |
| #define ON_LINE         0x10	/* status bit saying that printer is online */
 | |
| #define STATUS_MASK	0xB0	/* mask to filter out status bits */ 
 | |
| 
 | |
| #define MAX_ONLINE_RETRIES 120  /* about 60s: waits 0.5s after each retry */
 | |
| 
 | |
| /* Centronics interface timing that must be met by software (in microsec).
 | |
|  *
 | |
|  * Strobe length:	0.5u to 100u (not sure about the upper limit).
 | |
|  * Data set up:		0.5u before strobe.
 | |
|  * Data hold:		0.5u after strobe.
 | |
|  * Init pulse length:	over 200u (not sure).
 | |
|  *
 | |
|  * The strobe length is about 50u with the code here and function calls for
 | |
|  * sys_outb() - not much to spare.  The 0.5u minimums will not be violated 
 | |
|  * with the sys_outb() messages exchanged.
 | |
|  */
 | |
| 
 | |
| static endpoint_t caller;	/* process to tell when printing done (FS) */
 | |
| static int revive_pending;	/* set to true if revive is pending */
 | |
| static int revive_status;	/* revive status */
 | |
| static int done_status;	/* status of last output completion */
 | |
| static int oleft;		/* bytes of output left in obuf */
 | |
| static unsigned char obuf[128];	/* output buffer */
 | |
| static unsigned const char *optr;	/* ptr to next char in obuf to print */
 | |
| static int orig_count;		/* original byte count */
 | |
| static int port_base;		/* I/O port for printer */
 | |
| static endpoint_t proc_nr;	/* user requesting the printing */
 | |
| static cp_grant_id_t grant_nr;	/* grant on which print happens */
 | |
| static int user_left;		/* bytes of output left in user buf */
 | |
| static vir_bytes user_vir_d;	/* offset in user buf */
 | |
| int writing;		/* nonzero while write is in progress */
 | |
| static int irq_hook_id;	/* id of irq hook at kernel */
 | |
| 
 | |
| static void do_cancel(message *m_ptr);
 | |
| static void output_done(void);
 | |
| static void do_write(message *m_ptr);
 | |
| static void do_status(message *m_ptr);
 | |
| static void prepare_output(void);
 | |
| static int do_probe(void);
 | |
| static void do_initialize(void);
 | |
| static void reply(int code,int replyee,int proc,int status);
 | |
| static void do_printer_output(void);
 | |
| 
 | |
| /* SEF functions and variables. */
 | |
| static void sef_local_startup(void);
 | |
| static int sef_cb_init_fresh(int type, sef_init_info_t *info);
 | |
| EXTERN int sef_cb_lu_prepare(int state);
 | |
| EXTERN int sef_cb_lu_state_isvalid(int state);
 | |
| EXTERN void sef_cb_lu_state_dump(int state);
 | |
| int is_status_msg_expected = FALSE;
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				printer_task				     *
 | |
|  *===========================================================================*/
 | |
| int main(void)
 | |
| {
 | |
| /* Main routine of the printer task. */
 | |
|   message pr_mess;		/* buffer for all incoming messages */
 | |
|   int ipc_status;
 | |
| 
 | |
|   /* SEF local startup. */
 | |
|   sef_local_startup();
 | |
| 
 | |
|   while (TRUE) {
 | |
| 	if(driver_receive(ANY, &pr_mess, &ipc_status) != OK) {
 | |
| 		panic("driver_receive failed");
 | |
| 	}
 | |
| 
 | |
| 	if (is_ipc_notify(ipc_status)) {
 | |
| 		switch (_ENDPOINT_P(pr_mess.m_source)) {
 | |
| 			case HARDWARE:
 | |
| 				do_printer_output();
 | |
| 				break;
 | |
| 			default:
 | |
| 				reply(TASK_REPLY, pr_mess.m_source,
 | |
| 						pr_mess.USER_ENDPT, EINVAL);
 | |
| 		}
 | |
| 		continue;
 | |
| 	}
 | |
| 
 | |
| 	switch(pr_mess.m_type) {
 | |
| 	    case DEV_OPEN:
 | |
|                  do_initialize();		/* initialize */
 | |
| 	        /* fall through */
 | |
| 	    case DEV_CLOSE:
 | |
| 		reply(TASK_REPLY, pr_mess.m_source, pr_mess.USER_ENDPT, OK);
 | |
| 		break;
 | |
| 	    case DEV_WRITE_S:	do_write(&pr_mess);	break;
 | |
| 	    case DEV_STATUS:	do_status(&pr_mess);	break;
 | |
| 	    case CANCEL:	do_cancel(&pr_mess);	break;
 | |
| 	    default:
 | |
| 		reply(TASK_REPLY, pr_mess.m_source, pr_mess.USER_ENDPT,
 | |
| 			EINVAL);
 | |
| 	}
 | |
|   }
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *			       sef_local_startup			     *
 | |
|  *===========================================================================*/
 | |
| static void sef_local_startup()
 | |
| {
 | |
|   /* Register init callbacks. */
 | |
|   sef_setcb_init_fresh(sef_cb_init_fresh);
 | |
|   sef_setcb_init_lu(sef_cb_init_fresh);
 | |
|   sef_setcb_init_restart(sef_cb_init_fresh);
 | |
| 
 | |
|   /* Register live update callbacks. */
 | |
|   sef_setcb_lu_prepare(sef_cb_lu_prepare);
 | |
|   sef_setcb_lu_state_isvalid(sef_cb_lu_state_isvalid);
 | |
|   sef_setcb_lu_state_dump(sef_cb_lu_state_dump);
 | |
| 
 | |
|   /* Register signal callbacks. */
 | |
|   sef_setcb_signal_handler(sef_cb_signal_handler_term);
 | |
| 
 | |
|   /* Let SEF perform startup. */
 | |
|   sef_startup();
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *		            sef_cb_init_fresh                                *
 | |
|  *===========================================================================*/
 | |
| static int sef_cb_init_fresh(int UNUSED(type), sef_init_info_t *UNUSED(info))
 | |
| {
 | |
| /* Initialize the printer driver. */
 | |
| 
 | |
|   /* If no printer is present, do not start. */
 | |
|   if (!do_probe())
 | |
| 	return ENODEV;	/* arbitrary error code */
 | |
| 
 | |
|   /* Announce we are up! */
 | |
|   chardriver_announce();
 | |
| 
 | |
|   return OK;
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_write				     *
 | |
|  *===========================================================================*/
 | |
| static void do_write(m_ptr)
 | |
| register message *m_ptr;	/* pointer to the newly arrived message */
 | |
| {
 | |
| /* The printer is used by sending DEV_WRITE messages to it. Process one. */
 | |
| 
 | |
|     register int r = SUSPEND;
 | |
|     int retries;
 | |
|     u32_t status;
 | |
| 
 | |
|     /* Reject command if last write is not yet finished, the count is not
 | |
|      * positive, or the user address is bad.
 | |
|      */
 | |
|     if (writing)  			r = EIO;
 | |
|     else if (m_ptr->COUNT <= 0)  	r = EINVAL;
 | |
| 
 | |
|     /* Reply to FS, no matter what happened, possible SUSPEND caller. */
 | |
|     reply(TASK_REPLY, m_ptr->m_source, m_ptr->USER_ENDPT, r);
 | |
| 
 | |
|     /* If no errors occurred, continue printing with SUSPENDED caller.
 | |
|      * First wait until the printer is online to prevent stupid errors.
 | |
|      */
 | |
|     if (SUSPEND == r) { 	
 | |
| 	caller = m_ptr->m_source;
 | |
| 	proc_nr = m_ptr->USER_ENDPT;
 | |
| 	user_left = m_ptr->COUNT;
 | |
| 	orig_count = m_ptr->COUNT;
 | |
| 	user_vir_d = 0;				 	/* Offset. */
 | |
| 	writing = TRUE;
 | |
| 	grant_nr = (cp_grant_id_t) m_ptr->IO_GRANT;
 | |
| 
 | |
|         retries = MAX_ONLINE_RETRIES + 1;  
 | |
|         while (--retries > 0) {
 | |
|             if(sys_inb(port_base + 1, &status) != OK) {
 | |
| 		printf("printer: sys_inb of %x failed\n", port_base+1);
 | |
| 		panic("sys_inb failed");
 | |
| 	    }
 | |
|             if ((status & ON_LINE)) {		/* printer online! */
 | |
| 	        prepare_output();
 | |
| 	        do_printer_output();
 | |
| 	        return;
 | |
|             }
 | |
|             micro_delay(500000);		/* wait before retry */
 | |
|         }
 | |
|         /* If we reach this point, the printer was not online in time. */
 | |
|         done_status = status;
 | |
|         output_done();
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				output_done				     *
 | |
|  *===========================================================================*/
 | |
| static void output_done()
 | |
| {
 | |
| /* Previous chunk of printing is finished.  Continue if OK and more.
 | |
|  * Otherwise, reply to caller (FS).
 | |
|  */
 | |
|     register int status;
 | |
| 
 | |
|     if (!writing) return;	  	/* probably leftover interrupt */
 | |
|     if (done_status != OK) {      	/* printer error occurred */
 | |
|         status = EIO;
 | |
| 	if ((done_status & ON_LINE) == 0) { 
 | |
| 	    printf("Printer is not on line\n");
 | |
| 	} else if ((done_status & NO_PAPER)) { 
 | |
| 	    printf("Printer is out of paper\n");
 | |
| 	    status = EAGAIN;	
 | |
| 	} else {
 | |
| 	    printf("Printer error, status is 0x%02X\n", done_status);
 | |
| 	}
 | |
| 	/* Some characters have been printed, tell how many. */
 | |
| 	if (status == EAGAIN && user_left < orig_count) {
 | |
| 		status = orig_count - user_left;
 | |
| 	}
 | |
| 	oleft = 0;			/* cancel further output */
 | |
|     } 
 | |
|     else if (user_left != 0) {		/* not yet done, continue! */
 | |
| 	prepare_output();
 | |
| 	return;
 | |
|     } 
 | |
|     else {				/* done! report back to FS */
 | |
| 	status = orig_count;
 | |
|     }
 | |
|     is_status_msg_expected = TRUE;
 | |
|     revive_pending = TRUE;
 | |
|     revive_status = status;
 | |
|     notify(caller);
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_status				     *
 | |
|  *===========================================================================*/
 | |
| static void do_status(m_ptr)
 | |
| register message *m_ptr;	/* pointer to the newly arrived message */
 | |
| {
 | |
|   if (revive_pending) {
 | |
| 	m_ptr->m_type = DEV_REVIVE;		/* build message */
 | |
| 	m_ptr->REP_ENDPT = proc_nr;
 | |
| 	m_ptr->REP_STATUS = revive_status;
 | |
| 	m_ptr->REP_IO_GRANT = grant_nr;
 | |
| 
 | |
| 	writing = FALSE;			/* unmark event */
 | |
| 	revive_pending = FALSE;			/* unmark event */
 | |
|   } else {
 | |
| 	m_ptr->m_type = DEV_NO_STATUS;
 | |
| 	
 | |
| 	is_status_msg_expected = FALSE;
 | |
|   }
 | |
|   send(m_ptr->m_source, m_ptr);			/* send the message */
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_cancel				     *
 | |
|  *===========================================================================*/
 | |
| static void do_cancel(m_ptr)
 | |
| register message *m_ptr;	/* pointer to the newly arrived message */
 | |
| {
 | |
| /* Cancel a print request that has already started.  Usually this means that
 | |
|  * the process doing the printing has been killed by a signal.  It is not
 | |
|  * clear if there are race conditions.  Try not to cancel the wrong process,
 | |
|  * but rely on FS to handle the EINTR reply and de-suspension properly.
 | |
|  */
 | |
| 
 | |
|   if (writing && m_ptr->USER_ENDPT == proc_nr) {
 | |
| 	oleft = 0;		/* cancel output by interrupt handler */
 | |
| 	writing = FALSE;
 | |
| 	revive_pending = FALSE;
 | |
|   }
 | |
|   reply(TASK_REPLY, m_ptr->m_source, m_ptr->USER_ENDPT, EINTR);
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				reply					     *
 | |
|  *===========================================================================*/
 | |
| static void reply(code, replyee, process, status)
 | |
| int code;			/* TASK_REPLY or REVIVE */
 | |
| int replyee;			/* destination for message (normally FS) */
 | |
| int process;			/* which user requested the printing */
 | |
| int status;			/* number of  chars printed or error code */
 | |
| {
 | |
| /* Send a reply telling FS that printing has started or stopped. */
 | |
| 
 | |
|   message pr_mess;
 | |
| 
 | |
|   pr_mess.m_type = code;		/* TASK_REPLY or REVIVE */
 | |
|   pr_mess.REP_STATUS = status;		/* count or EIO */
 | |
|   pr_mess.REP_ENDPT = process;	/* which user does this pertain to */
 | |
|   send(replyee, &pr_mess);		/* send the message */
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_probe				     *
 | |
|  *===========================================================================*/
 | |
| static int do_probe(void)
 | |
| {
 | |
| /* See if there is a printer at all. */
 | |
| 
 | |
|   /* Get the base port for first printer. */
 | |
|   if(sys_readbios(LPT1_IO_PORT_ADDR, &port_base, LPT1_IO_PORT_SIZE) != OK) {
 | |
| 	panic("do_initialize: sys_readbios failed");
 | |
|   }
 | |
| 
 | |
|   /* If the port is zero, the parallel port is not available at all. */
 | |
|   return (port_base != 0);
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_initialize				     *
 | |
|  *===========================================================================*/
 | |
| static void do_initialize()
 | |
| {
 | |
| /* Set global variables and initialize the printer. */
 | |
|   static int initialized = FALSE;
 | |
|   if (initialized) return;
 | |
|   initialized = TRUE;
 | |
|   
 | |
|   if(sys_outb(port_base + 2, INIT_PRINTER) != OK) {
 | |
| 	printf("printer: sys_outb of %x failed\n", port_base+2);
 | |
| 	panic("do_initialize: sys_outb init failed");
 | |
|   }
 | |
|   micro_delay(1000000/20);	/* easily satisfies Centronics minimum */
 | |
|   if(sys_outb(port_base + 2, PR_SELECT) != OK) {
 | |
| 	printf("printer: sys_outb of %x failed\n", port_base+2);
 | |
| 	panic("do_initialize: sys_outb select failed");
 | |
|   }
 | |
|   irq_hook_id = 0;
 | |
|   if(sys_irqsetpolicy(PRINTER_IRQ, 0, &irq_hook_id) != OK ||
 | |
|      sys_irqenable(&irq_hook_id) != OK) {
 | |
| 	panic("do_initialize: irq enabling failed");
 | |
|   }
 | |
| }
 | |
| 
 | |
| /*==========================================================================*
 | |
|  *		    	      prepare_output				    *
 | |
|  *==========================================================================*/
 | |
| static void prepare_output()
 | |
| {
 | |
| /* Start next chunk of printer output. Fetch the data from user space. */
 | |
|   int s;
 | |
|   register int chunk;
 | |
| 
 | |
|   if ( (chunk = user_left) > sizeof obuf) chunk = sizeof obuf;
 | |
| 
 | |
|   s=sys_safecopyfrom(caller, grant_nr, user_vir_d, (vir_bytes) obuf,
 | |
| 	chunk);
 | |
| 
 | |
|   if(s != OK) {
 | |
|   	done_status = EFAULT;
 | |
|   	output_done();
 | |
|   	return;
 | |
|   }
 | |
|   optr = obuf;
 | |
|   oleft = chunk;
 | |
| }
 | |
| 
 | |
| /*===========================================================================*
 | |
|  *				do_printer_output				     *
 | |
|  *===========================================================================*/
 | |
| static void do_printer_output()
 | |
| {
 | |
| /* This function does the actual output to the printer. This is called on an
 | |
|  * interrupt message sent from the generic interrupt handler that 'forwards'
 | |
|  * interrupts to this driver. The generic handler did not reenable the printer
 | |
|  * IRQ yet! 
 | |
|  */
 | |
| 
 | |
|   u32_t status;
 | |
|   pvb_pair_t char_out[3];
 | |
| 
 | |
|   if (oleft == 0) {
 | |
| 	/* Nothing more to print.  Turn off printer interrupts in case they
 | |
| 	 * are level-sensitive as on the PS/2.  This should be safe even
 | |
| 	 * when the printer is busy with a previous character, because the
 | |
| 	 * interrupt status does not affect the printer.
 | |
| 	 */
 | |
| 	if(sys_outb(port_base + 2, PR_SELECT) != OK) {
 | |
| 		printf("printer: sys_outb of %x failed\n", port_base+2);
 | |
| 		panic("sys_outb failed");
 | |
| 	}
 | |
| 	if(sys_irqenable(&irq_hook_id) != OK) {
 | |
| 		panic("sys_irqenable failed");
 | |
| 	}
 | |
| 	return;
 | |
|   }
 | |
| 
 | |
|   do {
 | |
| 	/* Loop to handle fast (buffered) printers.  It is important that
 | |
| 	 * processor interrupts are not disabled here, just printer interrupts.
 | |
| 	 */
 | |
| 	if(sys_inb(port_base + 1, &status) != OK) {
 | |
| 		printf("printer: sys_inb of %x failed\n", port_base+1);
 | |
| 		panic("sys_inb failed");
 | |
| 	}
 | |
| 	if ((status & STATUS_MASK) == BUSY_STATUS) {
 | |
| 		/* Still busy with last output.  This normally happens
 | |
| 		 * immediately after doing output to an unbuffered or slow
 | |
| 		 * printer.  It may happen after a call from prepare_output or
 | |
| 		 * pr_restart, since they are not synchronized with printer
 | |
| 		 * interrupts.  It may happen after a spurious interrupt.
 | |
| 		 */
 | |
| 		if(sys_irqenable(&irq_hook_id) != OK) {
 | |
| 			panic("sys_irqenable failed");
 | |
| 		}
 | |
| 		return;
 | |
| 	}
 | |
| 	if ((status & STATUS_MASK) == NORMAL_STATUS) {
 | |
| 		/* Everything is all right.  Output another character. */
 | |
| 		pv_set(char_out[0], port_base, *optr);	
 | |
| 		optr++;
 | |
| 		pv_set(char_out[1], port_base+2, ASSERT_STROBE);
 | |
| 		pv_set(char_out[2], port_base+2, NEGATE_STROBE);
 | |
| 		if(sys_voutb(char_out, 3) != OK) {
 | |
| 			/* request series of port outb */
 | |
| 			panic("sys_voutb failed");
 | |
| 		}
 | |
| 
 | |
| 		user_vir_d++;
 | |
| 		user_left--;
 | |
| 	} else {
 | |
| 		/* Error.  This would be better ignored (treat as busy). */
 | |
| 		done_status = status;
 | |
| 		output_done();
 | |
| 		if(sys_irqenable(&irq_hook_id) != OK) {
 | |
| 			panic("sys_irqenable failed");
 | |
| 		}
 | |
| 		return;
 | |
| 	}
 | |
|   }
 | |
|   while (--oleft != 0);
 | |
| 
 | |
|   /* Finished printing chunk OK. */
 | |
|   done_status = OK;
 | |
|   output_done();
 | |
|   if(sys_irqenable(&irq_hook_id) != OK) {
 | |
| 	panic("sys_irqenable failed");
 | |
|   }
 | |
| }
 | |
| 
 |