pdc_console.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:16k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  *  linux/drivers/char/pdc_console.c
  3.  *
  4.  *  2001, Christoph Plattner
  5.  * 
  6.  *  Driver template was linux's serial.c
  7.  *
  8.  */
  9. static char *pdc_drv_version = "0.3";
  10. static char *pdc_drv_revdate = "2001-11-17";
  11. #define AUTHOR "christoph.plattner@gmx.at"
  12. #include <linux/config.h>
  13. #include <linux/version.h>
  14. #undef PDC_DRV_DEBUG
  15. #undef SERIAL_PARANOIA_CHECK
  16. #define CONFIG_SERIAL_NOPAUSE_IO
  17. #define SERIAL_DO_RESTART
  18. #define PDC_POLL_DELAY (30 * HZ / 1000)
  19. /*
  20.  * End of serial driver configuration section.
  21.  */
  22. #include <linux/module.h>
  23. #include <linux/types.h>
  24. #include <linux/serial.h>
  25. #include <linux/serialP.h>
  26. #include <linux/serial_reg.h>
  27. #include <asm/serial.h>
  28. #define LOCAL_VERSTRING ""
  29. #include <linux/errno.h>
  30. #include <linux/signal.h>
  31. #include <linux/sched.h>
  32. #include <linux/timer.h>
  33. #include <linux/interrupt.h>
  34. #include <linux/tty.h>
  35. #include <linux/tty_flip.h>
  36. #include <linux/major.h>
  37. #include <linux/string.h>
  38. #include <linux/fcntl.h>
  39. #include <linux/ptrace.h>
  40. #include <linux/ioport.h>
  41. #include <linux/mm.h>
  42. #include <linux/slab.h>
  43. #include <linux/init.h>
  44. #include <asm/uaccess.h>
  45. #include <linux/delay.h>
  46. #include <asm/system.h>
  47. #include <asm/io.h>
  48. #include <asm/irq.h>
  49. #include <asm/bitops.h>
  50. #ifdef CONFIG_GSC
  51. #include <asm/gsc.h>
  52. #endif
  53. extern int pdc_console_poll_key(void *);
  54. extern void pdc_outc(unsigned char);
  55. static char *pdc_drv_name = "PDC Software Console";
  56. static struct tty_driver pdc_drv_driver;
  57. static int pdc_drv_refcount = 0;
  58. static struct async_struct *pdc_drv_info;
  59. static struct timer_list pdc_drv_timer;
  60. /* serial subtype definitions */
  61. #ifndef SERIAL_TYPE_NORMAL
  62. #define SERIAL_TYPE_NORMAL 1
  63. #define SERIAL_TYPE_CALLOUT 2
  64. #endif
  65. #define NR_PORTS 1
  66. #define PDC_DUMMY_BUF 2048
  67. static struct tty_struct *pdc_drv_table[NR_PORTS];
  68. static struct termios *pdc_drv_termios[NR_PORTS];
  69. static struct termios *pdc_drv_termios_locked[NR_PORTS];
  70. /*
  71.  * tmp_buf is used as a temporary buffer by serial_write.  We need to
  72.  * lock it in case the copy_from_user blocks while swapping in a page,
  73.  * and some other program tries to do a serial write at the same time.
  74.  * Since the lock will only come under contention when the system is
  75.  * swapping and available memory is low, it makes sense to share one
  76.  * buffer across all the serial ports, since it significantly saves
  77.  * memory if large numbers of serial ports are open.
  78.  */
  79. static unsigned char *tmp_buf;
  80. #ifdef DECLARE_MUTEX
  81. static DECLARE_MUTEX(tmp_buf_sem);
  82. #else
  83. static struct semaphore tmp_buf_sem = MUTEX;
  84. #endif
  85. /*
  86.  * ------------------------------------------------------------
  87.  * pdc_stop() and pdc_start()
  88.  *
  89.  * This routines are called before setting or resetting tty->stopped.
  90.  * They enable or disable transmitter interrupts, as necessary.
  91.  * ------------------------------------------------------------
  92.  */
  93. static void
  94. pdc_stop(struct tty_struct *tty)
  95. {
  96. }
  97. static void
  98. pdc_start(struct tty_struct *tty)
  99. {
  100. }
  101. /*
  102.  * ----------------------------------------------------------------------
  103.  *
  104.  * Here starts the interrupt handling routines.  All of the following
  105.  * subroutines are declared as inline and are folded into
  106.  * rs_interrupt().  They were separated out for readability's sake.
  107.  *
  108.  * Note: rs_interrupt() is a "fast" interrupt, which means that it
  109.  * runs with interrupts turned off.  People who may want to modify
  110.  * rs_interrupt() should try to keep the interrupt handler as fast as
  111.  * possible.  After you are done making modifications, it is not a bad
  112.  * idea to do:
  113.  * 
  114.  * gcc -S -DKERNEL -Wall -Wstrict-prototypes -O6 -fomit-frame-pointer serial.c
  115.  *
  116.  * and look at the resulting assemble code in serial.s.
  117.  *
  118.  *  - Ted Ts'o (tytso@mit.edu), 7-Mar-93
  119.  * -----------------------------------------------------------------------
  120.  */
  121. static void
  122. receive_chars(struct async_struct *info, int *status, struct pt_regs *regs)
  123. {
  124. struct tty_struct *tty = info->tty;
  125. unsigned char ch;
  126. int __ch;
  127. while (1) {
  128. __ch = pdc_console_poll_key(NULL);
  129. if (__ch == -1) /* no character available */
  130. break;
  131. ch = (unsigned char) ((unsigned) __ch & 0x000000ffu);
  132. if (tty->flip.count >= TTY_FLIPBUF_SIZE)
  133. continue;
  134. *tty->flip.char_buf_ptr = ch;
  135. *tty->flip.flag_buf_ptr = 0;
  136. tty->flip.flag_buf_ptr++;
  137. tty->flip.char_buf_ptr++;
  138. tty->flip.count++;
  139. }
  140. tty_flip_buffer_push(tty);
  141. }
  142. static void
  143. pdc_drv_poll(unsigned long dummy)
  144. {
  145. struct async_struct *info;
  146. int status = 0;
  147. info = pdc_drv_info;
  148. if (!info || !info->tty || (pdc_drv_refcount == 0)) {
  149. /* do nothing */
  150. } else {
  151. receive_chars(info, &status, NULL);
  152. info->last_active = jiffies;
  153. }
  154. mod_timer(&pdc_drv_timer, jiffies + PDC_POLL_DELAY);
  155. }
  156. static void
  157. pdc_put_char(struct tty_struct *tty, unsigned char ch)
  158. {
  159. #ifdef PDC_DRV_DEBUG
  160. printk(KERN_NOTICE "[%s] %c returnn", __FUNCTION__, ch);
  161. #endif
  162. pdc_outc(ch);
  163. }
  164. static void
  165. pdc_flush_chars(struct tty_struct *tty)
  166. {
  167. /* PCD console always flushed all characters */
  168. #ifdef PDC_DRV_DEBUG
  169. printk(KERN_NOTICE "[%s] returnn", __FUNCTION__);
  170. #endif
  171. /* nothing to do */
  172. }
  173. static int
  174. pdc_write(struct tty_struct *tty, int from_user,
  175.   const unsigned char *buf, int count)
  176. {
  177. char pdc_tmp_buf[PDC_DUMMY_BUF];
  178. char *pdc_tmp_buf_ptr;
  179. int len;
  180. int ret = 0;
  181. #ifdef PDC_DRV_DEBUG
  182. printk(KERN_NOTICE "[%s] entryn", __FUNCTION__);
  183. #endif
  184. while (count) {
  185. if (count < PDC_DUMMY_BUF)
  186. len = count;
  187. else
  188. len = PDC_DUMMY_BUF;
  189. if (from_user) {
  190. copy_from_user(pdc_tmp_buf, buf, len);
  191. pdc_tmp_buf_ptr = pdc_tmp_buf;
  192. } else
  193. pdc_tmp_buf_ptr = (char *) buf;
  194. while (len) {
  195. pdc_outc(*pdc_tmp_buf_ptr);
  196. buf++;
  197. pdc_tmp_buf_ptr++;
  198. ret++;
  199. count--;
  200. len--;
  201. }
  202. }
  203. #ifdef PDC_DRV_DEBUG
  204. printk(KERN_NOTICE "[%s] returnn", __FUNCTION__);
  205. #endif
  206. return ret;
  207. }
  208. static int
  209. pdc_write_room(struct tty_struct *tty)
  210. {
  211. #ifdef PDC_DRV_DEBUG
  212. printk(KERN_NOTICE "[%s] entryn", __FUNCTION__);
  213. #endif
  214. return PDC_DUMMY_BUF;
  215. }
  216. static int
  217. pdc_chars_in_buffer(struct tty_struct *tty)
  218. {
  219. #ifdef PDC_DRV_DEBUG
  220. printk(KERN_NOTICE "[%s] entryn", __FUNCTION__);
  221. #endif
  222. return 0; /* no characters in buffer, always flushed ! */
  223. }
  224. static void
  225. pdc_flush_buffer(struct tty_struct *tty)
  226. {
  227. #ifdef PDC_DRV_DEBUG
  228. printk(KERN_NOTICE "[%s] returnn", __FUNCTION__);
  229. #endif
  230. }
  231. /*
  232.  * This function is used to send a high-priority XON/XOFF character to
  233.  * the device
  234.  */
  235. static void
  236. pdc_send_xchar(struct tty_struct *tty, char ch)
  237. {
  238. }
  239. /*
  240.  * ------------------------------------------------------------
  241.  * pdc_throttle()
  242.  * 
  243.  * This routine is called by the upper-layer tty layer to signal that
  244.  * incoming characters should be throttled.
  245.  * ------------------------------------------------------------
  246.  */
  247. static void
  248. pdc_throttle(struct tty_struct *tty)
  249. {
  250. }
  251. static void
  252. pdc_unthrottle(struct tty_struct *tty)
  253. {
  254. }
  255. /*
  256.  * ------------------------------------------------------------
  257.  * pdc_ioctl() and friends
  258.  * ------------------------------------------------------------
  259.  */
  260. static void
  261. pdc_break(struct tty_struct *tty, int break_state)
  262. {
  263. }
  264. static int
  265. get_serial_info(struct async_struct * info,
  266.                            struct serial_struct * retinfo)
  267. {
  268. struct serial_struct tmp;
  269. if (!retinfo)
  270. return -EFAULT;
  271. memset(&tmp, 0, sizeof(tmp));
  272. tmp.line = info->line;
  273. tmp.port = info->line;
  274. tmp.flags = info->flags;
  275. tmp.close_delay = info->close_delay;
  276. return copy_to_user(retinfo,&tmp,sizeof(*retinfo)) ? -EFAULT : 0;
  277. }
  278. static int get_modem_info(struct async_struct * info, unsigned int *value)
  279. {
  280. unsigned int result = TIOCM_DTR|TIOCM_CAR|TIOCM_CTS|TIOCM_RTS;
  281. return copy_to_user(value, &result, sizeof(int)) ? -EFAULT : 0;
  282. }
  283. static int get_lsr_info(struct async_struct * info, unsigned int *value)
  284. {
  285. unsigned int result = TIOCSER_TEMT;
  286. return copy_to_user(value, &result, sizeof(int)) ? -EFAULT : 0;
  287. }
  288. static int
  289. pdc_ioctl(struct tty_struct *tty, struct file *file,
  290.   unsigned int cmd, unsigned long arg)
  291. {
  292. struct async_struct *info = (struct async_struct *) tty->driver_data;
  293. if ((cmd != TIOCGSERIAL) && (cmd != TIOCSSERIAL) &&
  294.     (cmd != TIOCSERCONFIG) && (cmd != TIOCSERGSTRUCT) &&
  295.     (cmd != TIOCMIWAIT) && (cmd != TIOCGICOUNT)) {
  296. if (tty->flags & (1 << TTY_IO_ERROR))
  297. return -EIO;
  298. }
  299. switch (cmd) {
  300. case TIOCMGET:
  301. return get_modem_info(info, (unsigned int *) arg);
  302. case TIOCMBIS:
  303. case TIOCMBIC:
  304. case TIOCMSET:
  305. return 0;
  306. case TIOCGSERIAL:
  307. return get_serial_info(info, (struct serial_struct *) arg);
  308. case TIOCSSERIAL:
  309. return 0;
  310. case TIOCSERCONFIG:
  311. return 0;
  312. case TIOCSERGETLSR: /* Get line status register */
  313. return get_lsr_info(info, (unsigned int *) arg);
  314. case TIOCSERGSTRUCT:
  315. if (copy_to_user((struct async_struct *) arg,
  316.  info, sizeof (struct async_struct)))
  317. return -EFAULT;
  318. return 0;
  319. case TIOCMIWAIT:
  320. return 0;
  321. case TIOCGICOUNT:
  322. return 0;
  323. case TIOCSERGWILD:
  324. case TIOCSERSWILD:
  325. /* "setserial -W" is called in Debian boot */
  326. printk("TIOCSER?WILD ioctl obsolete, ignored.n");
  327. return 0;
  328. default:
  329. return -ENOIOCTLCMD;
  330. }
  331. return 0;
  332. }
  333. static void
  334. pdc_set_termios(struct tty_struct *tty, struct termios *old_termios)
  335. {
  336. #if 0 /* XXX CP, has to be checked, if there is stuff to do */
  337. struct async_struct *info = (struct async_struct *) tty->driver_data;
  338. unsigned long flags;
  339. unsigned int cflag = tty->termios->c_cflag;
  340. if ((cflag == old_termios->c_cflag)
  341.     && (RELEVANT_IFLAG(tty->termios->c_iflag)
  342. == RELEVANT_IFLAG(old_termios->c_iflag)))
  343. return;
  344. #if 0
  345. change_speed(info, old_termios);
  346. #endif
  347. /* Handle turning off CRTSCTS */
  348. if ((old_termios->c_cflag & CRTSCTS) &&
  349.     !(tty->termios->c_cflag & CRTSCTS)) {
  350. tty->hw_stopped = 0;
  351. pdc_start(tty);
  352. }
  353. #endif
  354. }
  355. /*
  356.  * ------------------------------------------------------------
  357.  * pdc_close()
  358.  * 
  359.  * This routine is called when the serial port gets closed.  First, we
  360.  * wait for the last remaining data to be sent.  Then, we unlink its
  361.  * async structure from the interrupt chain if necessary, and we free
  362.  * that IRQ if nothing is left in the chain.
  363.  * ------------------------------------------------------------
  364.  */
  365. static void
  366. pdc_close(struct tty_struct *tty, struct file *filp)
  367. {
  368. struct async_struct *info = (struct async_struct *) tty->driver_data;
  369. #ifdef PDC_DEBUG_OPEN
  370. printk("pdc_close ttyB%d, count = %dn", info->line, state->count);
  371. #endif
  372. pdc_drv_refcount--;
  373. if (pdc_drv_refcount > 0)
  374. return;
  375. info->flags |= ASYNC_CLOSING;
  376. /*
  377.  * Save the termios structure, since this port may have
  378.  * separate termios for callout and dialin.
  379.  */
  380. if (info->flags & ASYNC_NORMAL_ACTIVE)
  381. info->state->normal_termios = *tty->termios;
  382. if (info->flags & ASYNC_CALLOUT_ACTIVE)
  383. info->state->callout_termios = *tty->termios;
  384. /*
  385.  * At this point we stop accepting input.  To do this, we
  386.  * disable the receive line status interrupts, and tell the
  387.  * interrupt driver to stop checking the data ready bit in the
  388.  * line status register.
  389.  */
  390. /* XXX CP: make mask for receive !!! */
  391. if (tty->driver.flush_buffer)
  392. tty->driver.flush_buffer(tty);
  393. if (tty->ldisc.flush_buffer)
  394. tty->ldisc.flush_buffer(tty);
  395. tty->closing = 0;
  396. info->event = 0;
  397. info->tty = 0;
  398. pdc_drv_info = NULL;
  399. if (info->blocked_open) {
  400. if (info->close_delay) {
  401. set_current_state(TASK_INTERRUPTIBLE);
  402. schedule_timeout(info->close_delay);
  403. }
  404. wake_up_interruptible(&info->open_wait);
  405. }
  406. info->flags &= ~(ASYNC_NORMAL_ACTIVE | ASYNC_CALLOUT_ACTIVE |
  407.  ASYNC_CLOSING);
  408. wake_up_interruptible(&info->close_wait);
  409. MOD_DEC_USE_COUNT;
  410. }
  411. /*
  412.  * pdc_wait_until_sent() --- wait until the transmitter is empty
  413.  */
  414. static void
  415. pdc_wait_until_sent(struct tty_struct *tty, int timeout)
  416. {
  417. /* we always send immideate */
  418. }
  419. /*
  420.  * pdc_hangup() --- called by tty_hangup() when a hangup is signaled.
  421.  */
  422. static void
  423. pdc_hangup(struct tty_struct *tty)
  424. {
  425. }
  426. /*
  427.  * ------------------------------------------------------------
  428.  * pdc_open() and friends
  429.  * ------------------------------------------------------------
  430.  */
  431. static int
  432. get_async_struct(int line, struct async_struct **ret_info)
  433. {
  434. struct async_struct *info;
  435. info = kmalloc(sizeof (struct async_struct), GFP_KERNEL);
  436. if (!info) {
  437. return -ENOMEM;
  438. }
  439. memset(info, 0, sizeof (struct async_struct));
  440. init_waitqueue_head(&info->open_wait);
  441. init_waitqueue_head(&info->close_wait);
  442. init_waitqueue_head(&info->delta_msr_wait);
  443. info->magic = SERIAL_MAGIC;
  444. info->port = 0;
  445. info->flags = 0;
  446. info->io_type = 0;
  447. info->iomem_base = 0;
  448. info->iomem_reg_shift = 0;
  449. info->xmit_fifo_size = PDC_DUMMY_BUF;
  450. info->line = line;
  451. info->tqueue.routine = NULL;
  452. info->tqueue.data = info;
  453. info->state = NULL;
  454. *ret_info = info;
  455. return 0;
  456. }
  457. /*
  458.  * This routine is called whenever a serial port is opened.  It
  459.  * enables interrupts for a serial port, linking in its async structure into
  460.  * the IRQ chain.   It also performs the serial-specific
  461.  * initialization for the tty structure.
  462.  */
  463. static int
  464. pdc_open(struct tty_struct *tty, struct file *filp)
  465. {
  466. struct async_struct *info;
  467. int retval, line;
  468. unsigned long page;
  469. MOD_INC_USE_COUNT;
  470. line = MINOR(tty->device) - tty->driver.minor_start;
  471. if ((line < 0) || (line >= NR_PORTS)) {
  472. MOD_DEC_USE_COUNT;
  473. return -ENODEV;
  474. }
  475. retval = get_async_struct(line, &info);
  476. if (retval) {
  477. MOD_DEC_USE_COUNT;
  478. return retval;
  479. }
  480. tty->driver_data = info;
  481. info->tty = tty;
  482. pdc_drv_info = info;
  483. #ifdef PDC_DEBUG_OPEN
  484. printk("pdc_open %s%d, count = %dn", tty->driver.name, info->line,
  485.        info->state->count);
  486. #endif
  487. info->tty->low_latency = 0;
  488. if (!tmp_buf) {
  489. page = get_zeroed_page(GFP_KERNEL);
  490. if (!page) {
  491. MOD_DEC_USE_COUNT;
  492. return -ENOMEM;
  493. }
  494. if (tmp_buf)
  495. free_page(page);
  496. else
  497. tmp_buf = (unsigned char *) page;
  498. }
  499. info->session = current->session;
  500. info->pgrp = current->pgrp;
  501. #ifdef PDC_DEBUG_OPEN
  502. printk("pdc_open ttyB%d successful...", info->line);
  503. #endif
  504. pdc_drv_refcount++;
  505. return 0;
  506. }
  507. /*
  508.  * ---------------------------------------------------------------------
  509.  * pdc_init() and friends
  510.  *
  511.  * pdc_init() is called at boot-time to initialize the pdc driver.
  512.  * ---------------------------------------------------------------------
  513.  */
  514. static void
  515. show_pdc_drv_version(void)
  516. {
  517. printk(KERN_INFO "%s version %s%s (%s), %sn", pdc_drv_name,
  518.        pdc_drv_version, LOCAL_VERSTRING, pdc_drv_revdate, AUTHOR);
  519. }
  520. /*
  521.  * The serial driver boot-time initialization code!
  522.  */
  523. static int __init
  524. pdc_drv_init(void)
  525. {
  526. init_timer(&pdc_drv_timer);
  527. pdc_drv_timer.function = pdc_drv_poll;
  528. mod_timer(&pdc_drv_timer, jiffies + PDC_POLL_DELAY);
  529. show_pdc_drv_version();
  530. /* Initialize the tty_driver structure */
  531. memset(&pdc_drv_driver, 0, sizeof (struct tty_driver));
  532. pdc_drv_driver.magic = TTY_DRIVER_MAGIC;
  533. pdc_drv_driver.driver_name = "pdc_console";
  534. #ifdef CONFIG_DEVFS_FS
  535. pdc_drv_driver.name = "ttb/%d";
  536. #else
  537. pdc_drv_driver.name = "ttyB";
  538. #endif
  539. pdc_drv_driver.major = PDCCONS_MAJOR;
  540. pdc_drv_driver.minor_start = 0;
  541. pdc_drv_driver.num = NR_PORTS;
  542. pdc_drv_driver.type = TTY_DRIVER_TYPE_SERIAL;
  543. pdc_drv_driver.subtype = SERIAL_TYPE_NORMAL;
  544. pdc_drv_driver.init_termios = tty_std_termios;
  545. pdc_drv_driver.init_termios.c_cflag =
  546.     B9600 | CS8 | CREAD | HUPCL | CLOCAL;
  547. pdc_drv_driver.flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_NO_DEVFS;
  548. pdc_drv_driver.refcount = &pdc_drv_refcount;
  549. pdc_drv_driver.table = pdc_drv_table;
  550. pdc_drv_driver.termios = pdc_drv_termios;
  551. pdc_drv_driver.termios_locked = pdc_drv_termios_locked;
  552. pdc_drv_driver.open = pdc_open;
  553. pdc_drv_driver.close = pdc_close;
  554. pdc_drv_driver.write = pdc_write;
  555. pdc_drv_driver.put_char = pdc_put_char;
  556. pdc_drv_driver.flush_chars = pdc_flush_chars;
  557. pdc_drv_driver.write_room = pdc_write_room;
  558. pdc_drv_driver.chars_in_buffer = pdc_chars_in_buffer;
  559. pdc_drv_driver.flush_buffer = pdc_flush_buffer;
  560. pdc_drv_driver.ioctl = pdc_ioctl;
  561. pdc_drv_driver.throttle = pdc_throttle;
  562. pdc_drv_driver.unthrottle = pdc_unthrottle;
  563. pdc_drv_driver.set_termios = pdc_set_termios;
  564. pdc_drv_driver.stop = pdc_stop;
  565. pdc_drv_driver.start = pdc_start;
  566. pdc_drv_driver.hangup = pdc_hangup;
  567. pdc_drv_driver.break_ctl = pdc_break;
  568. pdc_drv_driver.send_xchar = pdc_send_xchar;
  569. pdc_drv_driver.wait_until_sent = pdc_wait_until_sent;
  570. pdc_drv_driver.read_proc = NULL;
  571. if (tty_register_driver(&pdc_drv_driver))
  572. panic("Couldn't register pdc_console drivern");
  573. return 0;
  574. }
  575. static void __exit
  576. pdc_fini(void)
  577. {
  578. int e1;
  579. if ((e1 = tty_unregister_driver(&pdc_drv_driver)))
  580. printk("pdc_console: failed to unregister pdc_drv driver (%d)n",
  581.        e1);
  582. }
  583. module_init(pdc_drv_init);
  584. module_exit(pdc_fini);
  585. MODULE_DESCRIPTION("PDC Software Console");
  586. MODULE_AUTHOR(AUTHOR);