如何查看Linux系统下串口信息?
从上面的信息可知串口设备为16550A,占用的io口为0x3f8。下面看下UART占用的端口的范围。
看到serial占用的io口范围为0x03f8~0x03ff,共8个port。
UART 16550A驱动代码8250_core.c文件中实现,初始化函数serial8250_init。从struct uart_driver类型的全局变量serial8250_reg来分析。
上面是用crash查看到的serial8250_reg的成员值,每个成员值的含义如下:
struct uart_driver { struct module *owner; const char *driver_name; const char *dev_name; int major; int minor; int nr; //该uart_driver支持的最大的串口个数 struct console *cons;
struct uart_state *state; //UART设备的状态 struct tty_driver *tty_driver; }; |
一个uart_driver对应多个uart_state,每一个uart_state对应一个uart_port,每个uart_port对应一个uart设备。
上图中看到的这个uart_port对应的device就是16550A设备
uart_port结构体的iobase成员是io端口的基地址,其值为0x3f8;serial_in成员函数是串口的读函数,对应为io_serial_in函数;serial_out成员函数是串口的写函数,对应为io_serial_out函数。前面说了这么多,就是为了看串口的读写函数。
static unsigned int io_serial_in(struct uart_port *p, int offset) { offset = offset << p->regshift; return inb(p->iobase + offset); }
static void io_serial_out(struct uart_port *p, int offset, int value) { offset = offset << p->regshift; outb(value, p->iobase + offset); } |
从上面两个函数可知如果我们要控制uart的端口达到我们的某些目标,iobase是已知且不变的,我们只需要按管脚来设置offset即可。
如果我们想操作RS232的第7个pin,也就是RTS,该怎么设置offset值?
查看UART 16550A手册即可知道RTS对应于Modem Control Register寄存器的bit1。而操作Modem Control Register寄存器的offset值为4。这个从内核的serial_reg.h文件中也可以查到。
因此如果想把pin7给拉高,写outb(0x20, p->iobase + 4)就行,想把pin7给拉低,写outb(0x0, p->iobase + 4)就行。