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

Linux/Unix编程

开发平台:

Unix_Linux

  1. #include <linux/console.h>
  2. #include <linux/kernel.h>
  3. #include <linux/init.h>
  4. #include <linux/string.h>
  5. #include <asm/io.h>
  6. #include <asm/proto.h>
  7. /* Simple VGA output */
  8. #define VGABASE 0xffffffff800b8000UL
  9. #define MAX_YPOS 25
  10. #define MAX_XPOS 80
  11. static int current_ypos = 1, current_xpos = 0; 
  12. static void early_vga_write(struct console *con, const char *str, unsigned n)
  13. {
  14. char c;
  15. int  i, k, j;
  16. while ((c = *str++) != '' && n-- > 0) {
  17. if (current_ypos >= MAX_YPOS) {
  18. /* scroll 1 line up */
  19. for(k = 1, j = 0; k < MAX_YPOS; k++, j++) {
  20. for(i = 0; i < MAX_XPOS; i++) {
  21. writew(readw(VGABASE + 2*(MAX_XPOS*k + i)),
  22.        VGABASE + 2*(MAX_XPOS*j + i));
  23. }
  24. }
  25. for(i = 0; i < MAX_XPOS; i++) {
  26. writew(0x720, VGABASE + 2*(MAX_XPOS*j + i));
  27. }
  28. current_ypos = MAX_YPOS-1;
  29. }
  30. if (c == 'n') {
  31. current_xpos = 0;
  32. current_ypos++;
  33. } else if (c != 'r')  {
  34. writew(((0x7 << 8) | (unsigned short) c),
  35.        VGABASE + 2*(MAX_XPOS*current_ypos + current_xpos++));
  36. if (current_xpos >= MAX_XPOS) {
  37. current_xpos = 0;
  38. current_ypos++;
  39. }
  40. }
  41. }
  42. }
  43. static struct console early_vga_console = {
  44. name: "earlyvga",
  45. write: early_vga_write,
  46. flags: CON_PRINTBUFFER,
  47. index: -1,
  48. };
  49. /* Serial functions losely based on a similar package from Klaus P. Gerlicher */ 
  50. int early_serial_base = 0x3f8;  /* ttyS0 */ 
  51. #define XMTRDY          0x20
  52. #define DLAB 0x80
  53. #define TXR             0       /*  Transmit register (WRITE) */
  54. #define RXR             0       /*  Receive register  (READ)  */
  55. #define IER             1       /*  Interrupt Enable          */
  56. #define IIR             2       /*  Interrupt ID              */
  57. #define FCR             2       /*  FIFO control              */
  58. #define LCR             3       /*  Line control              */
  59. #define MCR             4       /*  Modem control             */
  60. #define LSR             5       /*  Line Status               */
  61. #define MSR             6       /*  Modem Status              */
  62. #define DLL             0       /*  Divisor Latch Low         */
  63. #define DLH             1       /*  Divisor latch High        */
  64. static int early_serial_putc(unsigned char ch) 
  65. unsigned timeout = 0xffff; 
  66. while ((inb(early_serial_base + LSR) & XMTRDY) == 0 && --timeout) 
  67. rep_nop(); 
  68. outb(ch, early_serial_base + TXR);
  69. return timeout ? 0 : -1;
  70. static void early_serial_write(struct console *con, const char *s, unsigned n)
  71. {
  72. while (*s && n-- > 0) { 
  73. early_serial_putc(*s); 
  74. if (*s == 'n') 
  75. early_serial_putc('r'); 
  76. s++; 
  77. static __init void early_serial_init(char *opt)
  78. unsigned char c; 
  79. unsigned divisor, baud = 38400;
  80. char *s, *e;
  81. if (*opt == ',') 
  82. ++opt;
  83. s = strsep(&opt, ","); 
  84. if (s != NULL) { 
  85. unsigned port; 
  86. if (!strncmp(s,"0x",2))
  87. early_serial_base = simple_strtoul(s, &e, 16);
  88. else {
  89. static int bases[] = { 0x3f8, 0x2f8 };
  90. if (!strncmp(s,"ttyS",4)) 
  91. s+=4; 
  92. port = simple_strtoul(s, &e, 10); 
  93. if (port > 1 || s == e) 
  94. port = 0; 
  95. early_serial_base = bases[port];
  96. }
  97. outb(0x3, early_serial_base + LCR); /* 8n1 */
  98. outb(0, early_serial_base + IER); /* no interrupt */ 
  99. outb(0, early_serial_base + FCR); /* no fifo */ 
  100. outb(0x3, early_serial_base + MCR); /* DTR + RTS */ 
  101. s = strsep(&opt, ","); 
  102. if (s != NULL) { 
  103. baud = simple_strtoul(s, &e, 0); 
  104. if (baud == 0 || s == e) 
  105. baud = 38400;
  106. divisor = 115200 / baud; 
  107. c = inb(early_serial_base + LCR); 
  108. outb(c | DLAB, early_serial_base + LCR); 
  109. outb(divisor & 0xff, early_serial_base + DLL); 
  110. outb((divisor >> 8) & 0xff, early_serial_base + DLH); 
  111. outb(c & ~DLAB, early_serial_base + LCR);
  112. }
  113. static struct console early_serial_console = {
  114. name: "earlyser",
  115. write: early_serial_write,
  116. flags: CON_PRINTBUFFER,
  117. index: -1,
  118. };
  119. /* Direct interface for emergencies */
  120. struct console *early_console = &early_vga_console;
  121. static int early_console_initialized = 0;
  122. void early_printk(const char *fmt, ...)
  123. char buf[512]; 
  124. int n; 
  125. va_list ap;
  126. va_start(ap,fmt); 
  127. n = vsnprintf(buf,512,fmt,ap);
  128. early_console->write(early_console,buf,n);
  129. va_end(ap); 
  130. static int keep_early; 
  131. int __init setup_early_printk(char *opt) 
  132. {  
  133. char *space;
  134. char buf[256]; 
  135. if (early_console_initialized)
  136. return -1;
  137. strncpy(buf,opt,256); 
  138. buf[255] = 0; 
  139. space = strchr(buf, ' '); 
  140. if (space)
  141. *space = 0; 
  142. if (strstr(buf,"keep"))
  143. keep_early = 1; 
  144. if (!strncmp(buf, "serial", 6)) { 
  145. early_serial_init(buf + 6);
  146. early_console = &early_serial_console;
  147. } else if (!strncmp(buf, "vga", 3)) {
  148. early_console = &early_vga_console; 
  149. } else {
  150. early_console = NULL; 
  151. return -1; 
  152. }
  153. early_console_initialized = 1;
  154. register_console(early_console);       
  155. return 0;
  156. void __init disable_early_printk(void)
  157. if (!early_console_initialized || !early_console)
  158. return;
  159. if (!keep_early) {
  160. printk("Disabling early consolen");
  161. unregister_console(early_console);
  162. early_console_initialized = 0;
  163. /* syntax: earlyprintk=vga
  164.            earlyprintk=serial[,ttySn[,baudrate]] 
  165.    Append ,keep to not disable it when the real console takes over.
  166.    Only vga or serial at a time, not both.
  167.    Currently only ttyS0 and ttyS1 are supported. 
  168.    Interaction with the standard serial driver is not very good. 
  169.    The VGA output is eventually overwritten by the real console. */
  170. __setup("earlyprintk=", setup_early_printk);