日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > linux >内容正文

linux

慢慢欣赏linux之串口驱动代码分析 - 基于powerpc 2.6.x版本

發布時間:2024/3/26 linux 61 豆豆
生活随笔 收集整理的這篇文章主要介紹了 慢慢欣赏linux之串口驱动代码分析 - 基于powerpc 2.6.x版本 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

串口驅動分兩階段初始化

第一階段 ?串口驅動沒有初始化前的準備工作,包括設備樹的解析,platform設備注冊

asmlinkage void __init start_kernel(void)
?? ?=>setup_arch
?? ??? ?=>find_legacy_serial_ports();
?? ??? ??? ??? ?=>path = of_get_property(of_chosen, "linux,stdout-path", NULL);//解析設備樹,標記默認輸出串口,uboot可以設置該節點
?? ??? ??? ??? ?if (path != NULL) //path為 /soc4080@F BE00 0000/serial@11C500
?? ??? ??? ??? ??? ?stdout = of_find_node_by_path(path);//無stdout
?? ??? ??? ??? ?=>for_each_compatible_node(np, "serial", "ns16550")//解析并遍歷設備樹,填充2個結構體 legacy_serial_infos 和 legacy_serial_ports,第二階段需要
?? ??? ??? ??? ??? ?struct device_node *parent = of_get_parent(np);//對于powerpc p4080,串口的父節點是soc, 恰好匹配{.type = "soc",}
?? ??? ??? ??? ??? ?if (of_match_node(legacy_serial_parents, parent) != NULL) {
?? ??? ??? ??? ??? ??? ?index = add_legacy_soc_port(np, np);//從設備樹解析串口基本配置參數
?? ??? ??? ??? ??? ??? ??? ?=>of_get_property(np, "clock-frequency", NULL)
?? ??? ??? ??? ??? ??? ??? ?=>of_get_property(np, "reg-shift", NULL)
?? ??? ??? ??? ??? ??? ??? ?=>of_get_property(np, "reg-offset", NULL)
?? ??? ??? ??? ??? ??? ??? ?=>of_get_property(np, "used-by-rtas", NULL)
?? ??? ??? ??? ??? ??? ??? ?=>addrp = of_get_address(soc_dev, 0, NULL, NULL);
?? ??? ??? ??? ??? ??? ??? ?=>addr = of_translate_address(soc_dev, addrp);
?? ??? ??? ??? ??? ??? ??? ?=>add_legacy_port(np, -1, UPIO_MEM, addr, addr, NO_IRQ, flags, 0);
?? ??? ??? ??? ??? ??? ??? ??? ?=>clk = of_get_property(np, "clock-frequency", NULL);
?? ??? ??? ??? ??? ??? ??? ??? ?=>spd = of_get_property(np, "current-speed", NULL);
?? ??? ??? ??? ??? ??? ??? ??? ?=>if (want_index >= 0 && want_index < MAX_LEGACY_SERIAL_PORTS)
?? ??? ??? ??? ??? ??? ??? ??? ??? ?index = want_index;
?? ??? ??? ??? ??? ??? ??? ??? ?else
?? ??? ??? ??? ??? ??? ??? ??? ??? ?index = legacy_serial_count;
?? ??? ??? ??? ??? ??? ??? ??? ?=>if (index >= MAX_LEGACY_SERIAL_PORTS)
?? ??? ??? ??? ??? ??? ??? ??? ??? ?return -1;
?? ??? ??? ??? ??? ??? ??? ??? ?if (index >= legacy_serial_count)
?? ??? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_count = index + 1;
?? ??? ??? ??? ??? ??? ??? ??? ?=>memset(&legacy_serial_ports[index], 0, ?sizeof(struct plat_serial8250_port));
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_ports[index].mapbase = base;
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_ports[index].iotype = iotype;
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_ports[index].uartclk = clock;
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_ports[index].irq = irq;//這時中斷號為0,因為參數是NO_IRQ,后面會解析設備樹
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_ports[index].flags = flags;
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_infos[index].taddr = taddr;
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_infos[index].np = of_node_get(np);
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_infos[index].clock = clock;
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_infos[index].speed = spd ? *spd : 0;
?? ??? ??? ??? ??? ??? ??? ??? ?legacy_serial_infos[index].irq_check_parent = irq_check_parent;
?? ??? ??? ??? ??? ??? ??? ??? ?return index;

?? ??? ??? ??? ??? ??? ?if (index >= 0 && np == stdout)
?? ??? ??? ??? ??? ??? ??? ?legacy_serial_console = index;//標記默認串口,這個分支不會走
?? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ?of_node_put(parent);
?? ??? ??? ??? ?=>if (legacy_serial_console >= 0)//這個分支不會走
?? ??? ??? ??? ??? ?setup_legacy_serial_console(legacy_serial_console);
?? ?=>console_init();
?? ??? ?=>tty_ldisc_begin();//設置默認線路規程
?? ??? ??? ?=>tty_register_ldisc(N_TTY, &tty_ldisc_N_TTY);
?? ??? ??? ??? ?=>tty_ldiscs[disc] = new_ldisc;
?? ??? ??? ??? ?new_ldisc->num = disc;
?? ??? ??? ??? ?new_ldisc->refcount = 0;
?? ??? ?=>call = __con_initcall_start;
?? ??? ?=>while (call < __con_initcall_end) {
?? ??? ??? ?==>(*call)();
?? ??? ??? ??? ?=>console_initcall(serial8250_console_init);//初始化serial8250_ports,
?? ??? ??? ??? ??? ?數組大小 #define UART_NR?? ?CONFIG_SERIAL_8250_NR_UARTS
?? ??? ??? ??? ??? ?unsigned int nr_uarts = CONFIG_SERIAL_8250_RUNTIME_UARTS;
?? ??? ??? ??? ??? ?=>serial8250_isa_init_ports();
?? ??? ??? ??? ??? ??? ?=>struct uart_8250_port *up;
?? ??? ??? ??? ??? ??? ?static int first = 1;//這個函數只會進來一次,在這邊進來了,后面再也進不來了
?? ??? ??? ??? ??? ??? ?int i, irqflag = 0;

?? ??? ??? ??? ??? ??? ?if (!first)
?? ??? ??? ??? ??? ??? ??? ?return;
?? ??? ??? ??? ??? ??? ?first = 0;
?? ??? ??? ??? ??? ??? ?=>for (i = 0; i < nr_uarts; i++) {
?? ??? ??? ??? ??? ??? ??? ?struct uart_8250_port *up = &serial8250_ports[i];

?? ??? ??? ??? ??? ??? ??? ?up->port.line = i;
?? ??? ??? ??? ??? ??? ??? ?spin_lock_init(&up->port.lock);

?? ??? ??? ??? ??? ??? ??? ?init_timer(&up->timer);
?? ??? ??? ??? ??? ??? ??? ?up->timer.function = serial8250_timeout;

?? ??? ??? ??? ??? ??? ??? ?/*
?? ??? ??? ??? ??? ??? ??? ? * ALPHA_KLUDGE_MCR needs to be killed.
?? ??? ??? ??? ??? ??? ??? ? */
?? ??? ??? ??? ??? ??? ??? ?up->mcr_mask = ~ALPHA_KLUDGE_MCR;
?? ??? ??? ??? ??? ??? ??? ?up->mcr_force = ALPHA_KLUDGE_MCR;

?? ??? ??? ??? ??? ??? ??? ?up->port.ops = &serial8250_pops;
?? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ?=>后面的for循環在powerpc不會執行,因為在powerpc架構里面old_serial_port為空,而在x86則展開
?? ??? ??? ??? ??? ?=>register_console(&serial8250_console);
?? ??? ??? ??? ??? ??? ?static struct console serial8250_console = {
?? ??? ??? ??? ??? ??? ??? ?.name?? ??? ?= "ttyS",
?? ??? ??? ??? ??? ??? ??? ?.write?? ??? ?= serial8250_console_write,
?? ??? ??? ??? ??? ??? ??? ?.device?? ??? ?= uart_console_device,
?? ??? ??? ??? ??? ??? ??? ?.setup?? ??? ?= serial8250_console_setup,
?? ??? ??? ??? ??? ??? ??? ?.early_setup?? ?= serial8250_console_early_setup,
?? ??? ??? ??? ??? ??? ??? ?.flags?? ??? ?= CON_PRINTBUFFER,//只向緩沖區打印
?? ??? ??? ??? ??? ??? ??? ?.index?? ??? ?= -1,
?? ??? ??? ??? ??? ??? ??? ?.data?? ??? ?= &serial8250_reg,
?? ??? ??? ??? ??? ??? ?};
?? ??? ??? ??? ??? ??? ?=>if (newcon->early_setup)
?? ??? ??? ??? ??? ??? ??? ?newcon->early_setup();//static int serial8250_console_early_setup(void)
?? ??? ??? ??? ??? ??? ??? ??? ?=>return serial8250_find_port_for_earlycon();
?? ??? ??? ??? ??? ??? ??? ??? ??? ?=>line = serial8250_find_port(port);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>for (line = 0; line < nr_uarts; line++) {
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?port = &serial8250_ports[line].port;
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?if (uart_match_port(p, port))
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?return line;
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ??? ??? ??? ?=>ret = update_console_cmdline("uart", 8250, "ttyS", line, device->options);
?? ??? ??? ??? ??? ??? ?=>if ((newcon->flags & CON_CONSDEV) || console_drivers == NULL) {//將新的console加到鏈表里面, 也就是register的本質
?? ??? ??? ??? ??? ??? ??? ?newcon->next = console_drivers;
?? ??? ??? ??? ??? ??? ??? ?console_drivers = newcon;
?? ??? ??? ??? ??? ??? ??? ?if (newcon->next)
?? ??? ??? ??? ??? ??? ??? ??? ?newcon->next->flags &= ~CON_CONSDEV;
?? ??? ??? ??? ??? ??? ?} else {
?? ??? ??? ??? ??? ??? ??? ?newcon->next = console_drivers->next;
?? ??? ??? ??? ??? ??? ??? ?console_drivers->next = newcon;
?? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ?=>if (newcon->flags & CON_PRINTBUFFER) {
?? ??? ??? ??? ??? ??? ??? ?/*
?? ??? ??? ??? ??? ??? ??? ? * release_console_sem() will print out the buffered messages
?? ??? ??? ??? ??? ??? ??? ? * for us.
?? ??? ??? ??? ??? ??? ??? ? */
?? ??? ??? ??? ??? ??? ??? ?spin_lock_irqsave(&logbuf_lock, flags);
?? ??? ??? ??? ??? ??? ??? ?con_start = log_start;//打印buf的信息
?? ??? ??? ??? ??? ??? ??? ?spin_unlock_irqrestore(&logbuf_lock, flags);
?? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ?
?? ??? ??? ?==>call++;
?? ??? ?}
?? ?
int __init serial_dev_init(void)//注冊設備,根據設備樹,legacy_serial_infos 和 legacy_serial_ports 找中斷號
?? ?=>/*
?? ? * Before we register the platform serial devices, we need
?? ? * to fixup their interrupts and their IO ports.
?? ? */
?? ? for (i = 0; i < legacy_serial_count; i++) {
?? ??? ?struct plat_serial8250_port *port = &legacy_serial_ports[i];
?? ??? ?struct device_node *np = legacy_serial_infos[i].np;

?? ??? ?if (port->irq == NO_IRQ)
?? ??? ??? ?fixup_port_irq(i, np, port);
?? ??? ??? ??? ?=>virq = irq_of_parse_and_map(np, 0);//解析設備書找中斷號
?? ??? ??? ??? ?=>port->irq = virq;//找到中斷了
?? ??? ?if ((port->iotype == UPIO_MEM) || (port->iotype == UPIO_TSI))
?? ??? ??? ?fixup_port_mmio(i, np, port);
?? ??? ??? ??? ?=>port->membase = ioremap(port->mapbase, 0x100);
?? ?}
?? ?=>return platform_device_register(&serial_device);//注冊 serial_device platform設備,第二階段需要用
?? ??? ?static struct platform_device serial_device = {
?? ??? ?.name?? ?= "serial8250",
?? ??? ?.id?? ?= PLAT8250_DEV_PLATFORM,
?? ??? ?.dev?? ?= {
?? ??? ??? ?.platform_data = legacy_serial_ports,//legacy_serial_ports 在前面已經初始化
?? ??? ?},
?? ?};?? ?
?? ??? ?
第二階段 正式初始化
int __init serial8250_init(void)
?? ?=>serial8250_reg.nr = UART_NR;
?? ?ret = uart_register_driver(&serial8250_reg);//int uart_register_driver(struct uart_driver *drv)
?? ??? ?=>struct tty_driver *normal;
?? ??? ?=>drv->state = kzalloc(sizeof(struct uart_state) * drv->nr, GFP_KERNEL);
?? ??? ?=>normal = alloc_tty_driver(drv->nr);
?? ??? ?=>drv->tty_driver = normal; ?uart_driver 與 tty_driver 關聯
?? ??? ?=>normal->owner?? ??? ?= drv->owner;
?? ??? ?normal->driver_name?? ?= drv->driver_name;
?? ??? ?normal->name?? ??? ?= drv->dev_name;
?? ??? ?normal->major?? ??? ?= drv->major;
?? ??? ?normal->minor_start?? ?= drv->minor;
?? ??? ?normal->type?? ??? ?= TTY_DRIVER_TYPE_SERIAL;
?? ??? ?normal->subtype?? ??? ?= SERIAL_TYPE_NORMAL;
?? ??? ?normal->init_termios?? ?= tty_std_termios;
?? ??? ?normal->init_termios.c_cflag = B9600 | CS8 | CREAD | HUPCL | CLOCAL;
?? ??? ?normal->init_termios.c_ispeed = normal->init_termios.c_ospeed = 9600;
?? ??? ?normal->flags?? ??? ?= TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
?? ??? ?normal->driver_state ? ?= drv;tty_driver 與 uart_driver 關聯
?? ??? ?tty_set_operations(normal, &uart_ops);//void tty_set_operations(struct tty_driver *driver, const struct tty_operations *op)
?? ??? ??? ?=>driver->ops = op;
?? ??? ?=>/*
?? ??? ? * Initialise the UART state(s).
?? ??? ? */
?? ??? ? for (i = 0; i < drv->nr; i++) {
?? ??? ??? ?struct uart_state *state = drv->state + i;
?? ??? ??? ?struct tty_port *port = &state->port;

?? ??? ??? ?tty_port_init(port);
?? ??? ??? ?port->close_delay ? ? = 500;?? ?/* .5 seconds */
?? ??? ??? ?port->closing_wait ? ?= 30000;?? ?/* 30 seconds */
?? ??? ??? ?tasklet_init(&state->tlet, uart_tasklet_action,
?? ??? ??? ??? ??? ? (unsigned long)state);
?? ??? ?}
?? ??? ?=>retval = tty_register_driver(normal);
?? ??? ??? ?=>if (!(driver->flags & TTY_DRIVER_DEVPTS_MEM) && driver->num) {
?? ??? ??? ??? ?p = kzalloc(driver->num * 2 * sizeof(void *), GFP_KERNEL);
?? ??? ??? ??? ?if (!p)
?? ??? ??? ??? ??? ?return -ENOMEM;
?? ??? ??? ?}
?? ??? ??? ?=>if (!driver->major) {
?? ??? ??? ?error = alloc_chrdev_region(&dev, driver->minor_start,
?? ??? ??? ??? ??? ??? ??? ?driver->num, driver->name);
?? ??? ??? ?if (!error) {
?? ??? ??? ??? ?driver->major = MAJOR(dev);
?? ??? ??? ??? ?driver->minor_start = MINOR(dev);
?? ??? ??? ?}
?? ??? ?} else {
?? ??? ??? ?dev = MKDEV(driver->major, driver->minor_start);
?? ??? ??? ?error = register_chrdev_region(dev, driver->num, driver->name);
?? ??? ?}
?? ??? ?=>if (p) {
?? ??? ??? ?driver->ttys = (struct tty_struct **)p;
?? ??? ??? ?driver->termios = (struct ktermios **)(p + driver->num);
?? ??? ?}
?? ??? ?=>cdev_init(&driver->cdev, &tty_fops);
?? ??? ?driver->cdev.owner = driver->owner;
?? ??? ?error = cdev_add(&driver->cdev, dev, driver->num);
?? ??? ?=>list_add(&driver->tty_drivers, &tty_drivers);
?? ??? ?=>if (!(driver->flags & TTY_DRIVER_DYNAMIC_DEV)) {//這個流程不會進入,因為 TTY_DRIVER_DYNAMIC_DEV 被置
?? ??? ??? ?for (i = 0; i < driver->num; i++)
?? ??? ??? ??? ?tty_register_device(driver, i, NULL);
?? ??? ?}
?? ??? ?=>proc_tty_register_driver(driver);
?? ??? ?=>driver->flags |= TTY_DRIVER_INSTALLED;
?? ?=>serial8250_isa_devs = platform_device_alloc("serial8250", PLAT8250_DEV_LEGACY);
?? ?=>ret = platform_device_add(serial8250_isa_devs);//這個東東對x86有用,對于powerpc沒有太大用
?? ?=>serial8250_register_ports(&serial8250_reg, &serial8250_isa_devs->dev);
?? ?=>ret = platform_driver_register(&serial8250_isa_driver);//注冊 serial8250_isa_driver platform驅動, 與serial_device platform設備serial_device發生match, 調用probe函數
?? ??? ?static struct platform_driver serial8250_isa_driver = {
?? ??? ??? ?.probe?? ??? ?= serial8250_probe,
?? ??? ??? ?.remove?? ??? ?= __devexit_p(serial8250_remove),
?? ??? ??? ?.suspend?? ?= serial8250_suspend,
?? ??? ??? ?.resume?? ??? ?= serial8250_resume,
?? ??? ??? ?.driver?? ??? ?= {
?? ??? ??? ??? ?.name?? ?= "serial8250",
?? ??? ??? ??? ?.owner?? ?= THIS_MODULE,
?? ??? ??? ?},
?? ??? ?};

int __devinit serial8250_probe(struct platform_device *dev)
?? ?=>struct plat_serial8250_port *p = dev->dev.platform_data;
?? ?=>memset(&port, 0, sizeof(struct uart_port));
?? ?=>for (i = 0; p && p->flags != 0; p++, i++) {
?? ??? ?port.iobase?? ??? ?= p->iobase;
?? ??? ?port.membase?? ??? ?= p->membase;
?? ??? ?port.irq?? ??? ?= p->irq;
?? ??? ?port.irqflags?? ??? ?= p->irqflags;
?? ??? ?port.uartclk?? ??? ?= p->uartclk;
?? ??? ?port.regshift?? ??? ?= p->regshift;
?? ??? ?port.iotype?? ??? ?= p->iotype;
?? ??? ?port.flags?? ??? ?= p->flags;
?? ??? ?port.mapbase?? ??? ?= p->mapbase;
?? ??? ?port.hub6?? ??? ?= p->hub6;
?? ??? ?port.private_data?? ?= p->private_data;
?? ??? ?port.type?? ??? ?= p->type;
?? ??? ?port.serial_in?? ??? ?= p->serial_in;
?? ??? ?port.serial_out?? ??? ?= p->serial_out;
?? ??? ?port.dev?? ??? ?= &dev->dev;
?? ??? ?port.irqflags?? ??? ?|= irqflag;
?? ??? ?ret = serial8250_register_port(&port);
?? ??? ??? ?=>uart = serial8250_find_match_or_unused(port);
?? ??? ??? ?=>if (uart) {
?? ??? ??? ??? ?uart_remove_one_port(&serial8250_reg, &uart->port);

?? ??? ??? ??? ?uart->port.iobase ? ? ? = port->iobase;
?? ??? ??? ??? ?uart->port.membase ? ? ?= port->membase;
?? ??? ??? ??? ?uart->port.irq ? ? ? ? ?= port->irq;
?? ??? ??? ??? ?uart->port.irqflags ? ? = port->irqflags;
?? ??? ??? ??? ?uart->port.uartclk ? ? ?= port->uartclk;
?? ??? ??? ??? ?uart->port.fifosize ? ? = port->fifosize;
?? ??? ??? ??? ?uart->port.regshift ? ? = port->regshift;
?? ??? ??? ??? ?uart->port.iotype ? ? ? = port->iotype;
?? ??? ??? ??? ?uart->port.flags ? ? ? ?= port->flags | UPF_BOOT_AUTOCONF;
?? ??? ??? ??? ?uart->port.mapbase ? ? ?= port->mapbase;
?? ??? ??? ??? ?uart->port.private_data = port->private_data;
?? ??? ??? ??? ?if (port->dev)
?? ??? ??? ??? ??? ?uart->port.dev = port->dev;

?? ??? ??? ??? ?if (port->flags & UPF_FIXED_TYPE)
?? ??? ??? ??? ??? ?serial8250_init_fixed_type_port(uart, port->type);

?? ??? ??? ??? ?set_io_from_upio(&uart->port);
?? ??? ??? ??? ?/* Possibly override default I/O functions. ?*/
?? ??? ??? ??? ?if (port->serial_in)
?? ??? ??? ??? ??? ?uart->port.serial_in = port->serial_in;
?? ??? ??? ??? ?if (port->serial_out)
?? ??? ??? ??? ??? ?uart->port.serial_out = port->serial_out;

?? ??? ??? ??? ?ret = uart_add_one_port(&serial8250_reg, &uart->port);int uart_add_one_port(struct uart_driver *drv, struct uart_port *uport)
?? ??? ??? ??? ??? ?=>struct uart_state *state;
?? ??? ??? ??? ??? ?struct tty_port *port;
?? ??? ??? ??? ??? ?int ret = 0;
?? ??? ??? ??? ??? ?struct device *tty_dev;
?? ??? ??? ??? ??? ?=>state = drv->state + uport->line;line 從0開始
?? ??? ??? ??? ??? ?port = &state->port;
?? ??? ??? ??? ??? ?state->uart_port = uport; uart_port 與 uart_driver 關聯 ?( drv->state + uport->line)->uart_port = uport
?? ??? ??? ??? ??? ?state->pm_state = -1;

?? ??? ??? ??? ??? ?uport->cons = drv->cons;
?? ??? ??? ??? ??? ?uport->state = state; uart_driver 與 uart_port 關聯 uport->state = drv->state + uport->line
?? ??? ??? ??? ??? ?=>uart_configure_port(drv, state, uport);
?? ??? ??? ??? ??? ??? ?=>port->ops->config_port(port, flags);
?? ??? ??? ??? ??? ??? ?/\
?? ??? ??? ??? ??? ??? ?||
?? ??? ??? ??? ??? ??? ?V
?? ??? ??? ??? ??? ??? ?.config_port?? ?= serial8250_config_port//硬件寄存器配置都在這里面
?? ??? ??? ??? ??? ??? ??? ?=>autoconfig(up, probeflags);
?? ??? ??? ??? ??? ??? ??? ??? ?=>scratch = serial_in(up, UART_IIR) >> 6;
?? ??? ??? ??? ??? ??? ??? ??? ?=>switch (scratch)
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?case 3:
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?autoconfig_16550a(up);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?break;
?? ??? ??? ??? ??? ??? ?=>uart_report_port(drv, port);
?? ??? ??? ??? ??? ??? ?=>uart_change_pm(state, 0);/* Power up port for set_mctrl() */
?? ??? ??? ??? ??? ??? ?=>port->ops->set_mctrl(port, port->mctrl & TIOCM_DTR);
?? ??? ??? ??? ??? ??? ?=>register_console(port->cons);
?? ??? ??? ??? ??? ?=>tty_dev = tty_register_device(drv->tty_driver, uport->line, uport->dev);
?? ??? ??? ??? ??? ?=>if (likely(!IS_ERR(tty_dev))) {
?? ??? ??? ??? ??? ??? ?device_init_wakeup(tty_dev, 1);
?? ??? ??? ??? ??? ??? ?device_set_wakeup_enable(tty_dev, 0);
?? ??? ??? ??? ??? ?}
?? ??? ??? ??? ?if (ret == 0)
?? ??? ??? ??? ??? ?ret = uart->port.line;
?? ??? ??? ?}
?? ??? ?}
?? ?}
?? ?return 0;


內核態打印?? ??? ?
int printk(const char *fmt, ...)
?? ?=>r = vprintk(fmt, args);
?? ??? ?=>preempt_disable();//關閉搶占和關中斷
?? ??? ?raw_local_irq_save(flags);
?? ??? ?=>/* Emit the output into the temporary buffer *///準備好緩沖區
?? ??? ?printed_len += vscnprintf(printk_buf + printed_len,
?? ??? ??? ??? ??? ? ?sizeof(printk_buf) - printed_len, fmt, args);

?? ??? ?p = printk_buf;
?? ??? ?=>for ( ; *p; p++)//Copy the output into log_buf.
?? ??? ??? ??? ?emit_log_char('<');
?? ??? ??? ??? ?emit_log_char(current_log_level + '0');
?? ??? ??? ??? ?emit_log_char('>');
?? ??? ??? ??? ?if (!*p)
?? ??? ??? ??? ??? ?break;
?? ??? ??? ??? ?emit_log_char(*p);
?? ??? ??? ??? ?if (*p == '\n')
?? ??? ??? ??? ??? ?new_text_line = 1;
?? ??? ?=>if (acquire_console_semaphore_for_printk(this_cpu))
?? ??? ??? ?release_console_sem();
?? ??? ??? ??? ?=>for ( ; ; ) {//串口打印
?? ??? ??? ??? ??? ?spin_lock_irqsave(&logbuf_lock, flags);
?? ??? ??? ??? ??? ?wake_klogd |= log_start - log_end;
?? ??? ??? ??? ??? ?if (con_start == log_end)
?? ??? ??? ??? ??? ??? ?break;?? ??? ??? ?/* Nothing to print */
?? ??? ??? ??? ??? ?_con_start = con_start;
?? ??? ??? ??? ??? ?_log_end = log_end;
?? ??? ??? ??? ??? ?con_start = log_end;?? ??? ?/* Flush */
?? ??? ??? ??? ??? ?spin_unlock(&logbuf_lock);
?? ??? ??? ??? ??? ?stop_critical_timings();?? ?/* don't trace print latency */
?? ??? ??? ??? ??? ?call_console_drivers(_con_start, _log_end);
?? ??? ??? ??? ??? ??? ?=>_call_console_drivers(start_print, cur_index, msg_level);
?? ??? ??? ??? ??? ??? ??? ?=>if ((msg_log_level < console_loglevel || ignore_loglevel) &&
?? ??? ??? ??? ??? ??? ??? ??? ??? ?console_drivers && start != end) { //根據打印級別console_loglevel決定什么信息打印到串口上
?? ??? ??? ??? ??? ??? ??? ??? ?if ((start & LOG_BUF_MASK) > (end & LOG_BUF_MASK)) {
?? ??? ??? ??? ??? ??? ??? ??? ??? ?/* wrapped write */
?? ??? ??? ??? ??? ??? ??? ??? ??? ?__call_console_drivers(start & LOG_BUF_MASK, log_buf_len);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>con->write(con, &LOG_BUF(start), end - start);//對于第一階段來說,是 serial8250_console_write
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>struct uart_8250_port *up = &serial8250_ports[co->index];
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>ier = serial_in(up, UART_IER);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>uart_console_write(&up->port, s, count, serial8250_console_putchar);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>putchar(port, *s);//serial8250_console_putchar
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>struct uart_8250_port *up = (struct uart_8250_port *)port;
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>wait_for_xmitr(up, UART_LSR_THRE);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>serial_out(up, UART_TX, ch);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>wait_for_xmitr(up, BOTH_EMPTY);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>serial_out(up, UART_IER, ier);
?? ??? ??? ??? ??? ??? ??? ??? ??? ?__call_console_drivers(0, end & LOG_BUF_MASK);
?? ??? ??? ??? ??? ??? ??? ??? ?} else {
?? ??? ??? ??? ??? ??? ??? ??? ??? ?__call_console_drivers(start, end);
?? ??? ??? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ?start_critical_timings();
?? ??? ??? ??? ??? ?local_irq_restore(flags);
?? ??? ??? ??? ?}
?? ??? ??? ??? ?=>if (wake_klogd)//喚醒klogd,后續可以提交給syslogd打印到syslog或者messages里面
?? ??? ??? ??? ??? ?wake_up_klogd();
?? ??? ?=>raw_local_irq_restore(flags);
?? ??? ?preempt_enable();
?? ??? ??? ?
第二階段串口初始化完畢后
打開串口的代碼調用關系如下
tty_open
chrdev_open
__dentry_open
do_last
do_filp_open
do_sys_open
ret_from_syscall

int tty_open(struct inode *inode, struct file *filp)
?? ?=>dev_t device = inode->i_rdev;
?? ?=>driver = get_tty_driver(device, &index);
?? ?=>if (!tty)
?? ??? ?tty = tty_driver_lookup_tty(driver, inode, index);
?? ?=>if (tty) {
?? ??? ?retval = tty_reopen(tty);
?? ??? ?if (retval)
?? ??? ??? ?tty = ERR_PTR(retval);
?? ?} else
?? ??? ?tty = tty_init_dev(driver, index, 0);//struct tty_struct *tty_init_dev(struct tty_driver *driver, int idx, int first_ok)
?? ??? ??? ?=>struct tty_struct *tty
?? ??? ??? ?=>tty = alloc_tty_struct();
?? ??? ??? ?=>initialize_tty_struct(tty, driver, idx);
?? ??? ??? ??? ?=>tty->magic = TTY_MAGIC;
?? ??? ??? ??? ?tty_ldisc_init(tty);
?? ??? ??? ??? ?tty->buf.head = tty->buf.tail = NULL;
?? ??? ??? ??? ?tty_buffer_init(tty);
?? ??? ??? ??? ?tty->driver = driver;
?? ??? ??? ??? ?tty->ops = driver->ops;
?? ??? ??? ??? ?tty->index = idx;
?? ??? ??? ??? ?tty_line_name(driver, idx, tty->name);
?? ??? ??? ?=>retval = tty_driver_install_tty(driver, tty);
?? ??? ??? ?=>retval = tty_ldisc_setup(tty, tty->link);
?? ??? ??? ??? ?=>struct tty_ldisc *ld = tty->ldisc;
?? ??? ??? ??? ?=>retval = tty_ldisc_open(tty, ld);
?? ??? ??? ??? ?=>if (o_tty)?
?? ??? ??? ??? ??? ?retval = tty_ldisc_open(o_tty, o_tty->ldisc);
?? ??? ??? ??? ??? ?tty_ldisc_enable(o_tty);
?? ??? ??? ??? ?=>tty_ldisc_enable(tty);
?? ?=>filp->private_data = tty;
?? ?=>if (!retval) {
?? ??? ?if (tty->ops->open)
?? ??? ??? ?retval = tty->ops->open(tty, filp);//根據inode找到主從設備號,根據主從設備號從tty_drivers鏈表找到tty_driver
?? ??? ??? ?^
?? ??? ??? ?||
?? ??? ??? ?v
?? ??? ??? ?uart_open
?? ??? ??? ??? ?=>struct uart_driver *drv = (struct uart_driver *)tty->driver->driver_state;
?? ??? ??? ??? ?struct uart_state *state;
?? ??? ??? ??? ?struct tty_port *port;
?? ??? ??? ??? ?int retval, line = tty->index;
?? ??? ??? ??? ?=>state = uart_get(drv, line);
?? ??? ??? ??? ??? ?=>state = drv->state + line;
?? ??? ??? ??? ??? ?=>port = &state->port;
?? ??? ??? ??? ??? ?=>port->count++;
?? ??? ??? ??? ??? ?=>return state;
?? ??? ??? ??? ?=>port = &state->port;
?? ??? ??? ??? ?=>tty->driver_data = state;
?? ??? ??? ??? ?state->uart_port->state = state;
?? ??? ??? ??? ?tty->low_latency = (state->uart_port->flags & UPF_LOW_LATENCY) ? 1 : 0;
?? ??? ??? ??? ?tty->alt_speed = 0;
?? ??? ??? ??? ?tty_port_tty_set(port, tty);
?? ??? ??? ??? ?=>if (port->count == 1)
?? ??? ??? ??? ??? ?uart_change_pm(state, 0);
?? ??? ??? ??? ?=>retval = uart_startup(state, 0);
?? ??? ??? ??? ??? ?=>if (!state->xmit.buf) {
?? ??? ??? ??? ??? ??? ?/* This is protected by the per port mutex */
?? ??? ??? ??? ??? ??? ?page = get_zeroed_page(GFP_KERNEL);
?? ??? ??? ??? ??? ??? ?if (!page)
?? ??? ??? ??? ??? ??? ??? ?return -ENOMEM;

?? ??? ??? ??? ??? ??? ?state->xmit.buf = (unsigned char *) page;
?? ??? ??? ??? ??? ??? ?uart_circ_clear(&state->xmit);
?? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ?=>retval = uport->ops->startup(uport);
?? ??? ??? ??? ??? ?^
?? ??? ??? ??? ??? ?||
?? ??? ??? ??? ??? ?v
?? ??? ??? ??? ??? ?serial8250_startup
?? ??? ??? ??? ??? ??? ?=>struct uart_port *uport = state->uart_port;
?? ??? ??? ??? ??? ??? ?struct tty_port *port = &state->port;
?? ??? ??? ??? ??? ??? ?unsigned long page;
?? ??? ??? ??? ??? ??? ?=>if (!state->xmit.buf) {
?? ??? ??? ??? ??? ??? ??? ?/* This is protected by the per port mutex */
?? ??? ??? ??? ??? ??? ??? ?page = get_zeroed_page(GFP_KERNEL);
?? ??? ??? ??? ??? ??? ??? ?if (!page)
?? ??? ??? ??? ??? ??? ??? ??? ?return -ENOMEM;

?? ??? ??? ??? ??? ??? ??? ?state->xmit.buf = (unsigned char *) page;
?? ??? ??? ??? ??? ??? ??? ?uart_circ_clear(&state->xmit);
?? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ?=>retval = uport->ops->startup(uport);
?? ??? ??? ??? ??? ??? ?^
?? ??? ??? ??? ??? ??? ?||
?? ??? ??? ??? ??? ??? ?v
?? ??? ??? ??? ??? ??? ?static int serial8250_startup(struct uart_port *port)
?? ??? ??? ??? ??? ??? ??? ?=>struct uart_8250_port *up = (struct uart_8250_port *)port;
?? ??? ??? ??? ??? ??? ??? ?=>(void) serial_inp(up, UART_LSR);//先關閉硬件中斷源
?? ??? ??? ??? ??? ??? ??? ?(void) serial_inp(up, UART_RX);
?? ??? ??? ??? ??? ??? ??? ?(void) serial_inp(up, UART_IIR);
?? ??? ??? ??? ??? ??? ??? ?(void) serial_inp(up, UART_MSR);
?? ??? ??? ??? ??? ??? ??? ?=>spin_lock_irqsave(&up->port.lock, flags);//使能中斷控制器
?? ??? ??? ??? ??? ??? ??? ?if (up->port.irqflags & IRQF_SHARED)
?? ??? ??? ??? ??? ??? ??? ??? ??? ?enable_irq(up->port.irq);
?? ??? ??? ??? ??? ??? ??? ?spin_unlock_irqrestore(&up->port.lock, flags);
?? ??? ??? ??? ??? ??? ??? ?=>retval = serial_link_irq_chain(up);
?? ??? ??? ??? ??? ??? ??? ??? ?=>ret = request_irq(up->port.irq, serial8250_interrupt, irq_flags, "serial", i);//掛接中斷處理例程
?? ??? ??? ??? ??? ??? ??? ?=>spin_lock_irqsave(&up->port.lock, flags);//重新使能中斷源
?? ??? ??? ??? ??? ??? ??? ?/*
?? ??? ??? ??? ??? ??? ??? ? * Do a quick test to see if we receive an
?? ??? ??? ??? ??? ??? ??? ? * interrupt when we enable the TX irq.
?? ??? ??? ??? ??? ??? ??? ? */
?? ??? ??? ??? ??? ??? ??? ?serial_outp(up, UART_IER, UART_IER_THRI);
?? ??? ??? ??? ??? ??? ??? ?lsr = serial_in(up, UART_LSR);
?? ??? ??? ??? ??? ??? ??? ?iir = serial_in(up, UART_IIR);
?? ??? ??? ??? ??? ??? ??? ?serial_outp(up, UART_IER, 0);
?? ??? ??? ??? ??? ??? ??? ?spin_unlock_irqrestore(&up->port.lock, flags);
?? ??? ??? ??? ??? ??? ??? ?=>關閉再打開
?? ??? ??? ??? ??? ??? ??? ?/*
?? ??? ??? ??? ??? ??? ??? ? * Clear the interrupt registers again for luck, and clear the
?? ??? ??? ??? ??? ??? ??? ? * saved flags to avoid getting false values from polling
?? ??? ??? ??? ??? ??? ??? ? * routines or the previous session.
?? ??? ??? ??? ??? ??? ??? ? */
?? ??? ??? ??? ??? ??? ??? ?serial_inp(up, UART_LSR);
?? ??? ??? ??? ??? ??? ??? ?serial_inp(up, UART_RX);
?? ??? ??? ??? ??? ??? ??? ?serial_inp(up, UART_IIR);
?? ??? ??? ??? ??? ??? ??? ?serial_inp(up, UART_MSR);
?? ??? ??? ??? ??? ??? ??? ?up->lsr_saved_flags = 0;
?? ??? ??? ??? ??? ??? ??? ?up->msr_saved_flags = 0;

?? ??? ??? ??? ??? ??? ??? ?/*
?? ??? ??? ??? ??? ??? ??? ? * Finally, enable interrupts. ?Note: Modem status interrupts
?? ??? ??? ??? ??? ??? ??? ? * are set via set_termios(), which will be occurring imminently
?? ??? ??? ??? ??? ??? ??? ? * anyway, so we don't enable them here.
?? ??? ??? ??? ??? ??? ??? ? */
?? ??? ??? ??? ??? ??? ??? ?up->ier = UART_IER_RLSI | UART_IER_RDI;
?? ??? ??? ??? ??? ??? ??? ?serial_outp(up, UART_IER, up->ier);

?? ??? ??? ??? ??? ??? ??? ?if (up->port.flags & UPF_FOURPORT) {
?? ??? ??? ??? ??? ??? ??? ??? ?unsigned int icp;
?? ??? ??? ??? ??? ??? ??? ??? ?/*
?? ??? ??? ??? ??? ??? ??? ??? ? * Enable interrupts on the AST Fourport board
?? ??? ??? ??? ??? ??? ??? ??? ? */
?? ??? ??? ??? ??? ??? ??? ??? ?icp = (up->port.iobase & 0xfe0) | 0x01f;
?? ??? ??? ??? ??? ??? ??? ??? ?outb_p(0x80, icp);
?? ??? ??? ??? ??? ??? ??? ??? ?(void) inb_p(icp);
?? ??? ??? ??? ??? ??? ??? ?}

寫操作串口調用棧如下
uart_start
uart_write
n_tty_write
tty_write
vfs_write
sys_write
ret_from_syscall

輸出的代碼調用關系如下
ssize_t tty_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos)
?? ?=>struct tty_struct *tty;
?? ?struct inode *inode = file->f_path.dentry->d_inode;
?? ?struct tty_ldisc *ld;
?? ?=>tty = (struct tty_struct *)file->private_data;
?? ?ld = tty_ldisc_ref_wait(tty);
?? ?=>do_tty_write(ld->ops->write, tty, file, buf, count);
?? ??? ?for (;;)
?? ??? ??? ?copy_from_user(tty->write_buf, buf, size)
?? ??? ??? ?ret = write(tty, file, tty->write_buf, size)
?? ??? ??? ?^
?? ??? ??? ?||
?? ??? ??? ?v
?? ??? ??? ?ld->ops->write
?? ??? ??? ?^
?? ??? ??? ?||
?? ??? ??? ?v
?? ??? ??? ?n_tty_write
?? ??? ??? ??? ?struct tty_ldisc_ops tty_ldisc_N_TTY = {
?? ??? ??? ??? ??? ?.magic ? ? ? ? ? = TTY_LDISC_MAGIC,
?? ??? ??? ??? ??? ?.name ? ? ? ? ? ?= "n_tty",
?? ??? ??? ??? ??? ?.open ? ? ? ? ? ?= n_tty_open,
?? ??? ??? ??? ??? ?.close ? ? ? ? ? = n_tty_close,
?? ??? ??? ??? ??? ?.flush_buffer ? ?= n_tty_flush_buffer,
?? ??? ??? ??? ??? ?.chars_in_buffer = n_tty_chars_in_buffer,
?? ??? ??? ??? ??? ?.read ? ? ? ? ? ?= n_tty_read,
?? ??? ??? ??? ??? ?.write ? ? ? ? ? = n_tty_write,
?? ??? ??? ??? ??? ?.ioctl ? ? ? ? ? = n_tty_ioctl,
?? ??? ??? ??? ??? ?.set_termios ? ? = n_tty_set_termios,
?? ??? ??? ??? ??? ?.poll ? ? ? ? ? ?= n_tty_poll,
?? ??? ??? ??? ??? ?.receive_buf ? ? = n_tty_receive_buf,
?? ??? ??? ??? ??? ?.write_wakeup ? ?= n_tty_write_wakeup
?? ??? ??? ??? ?};
?? ??? ??? ??? ?=>const unsigned char *b = buf;
?? ??? ??? ??? ?DECLARE_WAITQUEUE(wait, current);
?? ??? ??? ??? ?int c;
?? ??? ??? ??? ?=>process_echoes(tty);
?? ??? ??? ??? ?=>add_wait_queue(&tty->write_wait, &wait);
?? ??? ??? ??? ?=>while (1)?
?? ??? ??? ??? ??? ?==>set_current_state(TASK_INTERRUPTIBLE);
?? ??? ??? ??? ??? ?if (signal_pending(current)) {
?? ??? ??? ??? ??? ??? ?retval = -ERESTARTSYS;
?? ??? ??? ??? ??? ??? ?break;
?? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ?==>while (nr > 0) {
?? ??? ??? ??? ??? ??? ?===>c = tty->ops->write(tty, b, nr);
?? ??? ??? ??? ??? ??? ?^
?? ??? ??? ??? ??? ??? ?||
?? ??? ??? ??? ??? ??? ?v
?? ??? ??? ??? ??? ??? ?uart_write
?? ??? ??? ??? ??? ??? ??? ?const struct tty_operations uart_ops = {
?? ??? ??? ??? ??? ??? ??? ??? ?.open?? ??? ?= uart_open,
?? ??? ??? ??? ??? ??? ??? ??? ?.close?? ??? ?= uart_close,
?? ??? ??? ??? ??? ??? ??? ??? ?.write?? ??? ?= uart_write,
?? ??? ??? ??? ??? ??? ??? ??? ?.put_char?? ?= uart_put_char,
?? ??? ??? ??? ??? ??? ??? ??? ?.flush_chars?? ?= uart_flush_chars,
?? ??? ??? ??? ??? ??? ??? ??? ?.write_room?? ?= uart_write_room,
?? ??? ??? ??? ??? ??? ??? ??? ?.chars_in_buffer= uart_chars_in_buffer,
?? ??? ??? ??? ??? ??? ??? ??? ?.flush_buffer?? ?= uart_flush_buffer,
?? ??? ??? ??? ??? ??? ??? ??? ?.ioctl?? ??? ?= uart_ioctl,
?? ??? ??? ??? ??? ??? ??? ??? ?.throttle?? ?= uart_throttle,
?? ??? ??? ??? ??? ??? ??? ??? ?.unthrottle?? ?= uart_unthrottle,
?? ??? ??? ??? ??? ??? ??? ??? ?.send_xchar?? ?= uart_send_xchar,
?? ??? ??? ??? ??? ??? ??? ??? ?.set_termios?? ?= uart_set_termios,
?? ??? ??? ??? ??? ??? ??? ??? ?.set_ldisc?? ?= uart_set_ldisc,
?? ??? ??? ??? ??? ??? ??? ??? ?.stop?? ??? ?= uart_stop,
?? ??? ??? ??? ??? ??? ??? ??? ?.start?? ??? ?= uart_start,
?? ??? ??? ??? ??? ??? ??? ??? ?.hangup?? ??? ?= uart_hangup,
?? ??? ??? ??? ??? ??? ??? ??? ?.break_ctl?? ?= uart_break_ctl,
?? ??? ??? ??? ??? ??? ??? ??? ?.wait_until_sent= uart_wait_until_sent,
?? ??? ??? ??? ??? ??? ??? ?#ifdef CONFIG_PROC_FS
?? ??? ??? ??? ??? ??? ??? ??? ?.proc_fops?? ?= &uart_proc_fops,
?? ??? ??? ??? ??? ??? ??? ?#endif
?? ??? ??? ??? ??? ??? ??? ??? ?.tiocmget?? ?= uart_tiocmget,
?? ??? ??? ??? ??? ??? ??? ??? ?.tiocmset?? ?= uart_tiocmset,
?? ??? ??? ??? ??? ??? ??? ?#ifdef CONFIG_CONSOLE_POLL
?? ??? ??? ??? ??? ??? ??? ??? ?.poll_init?? ?= uart_poll_init,
?? ??? ??? ??? ??? ??? ??? ??? ?.poll_get_char?? ?= uart_poll_get_char,
?? ??? ??? ??? ??? ??? ??? ??? ?.poll_put_char?? ?= uart_poll_put_char,
?? ??? ??? ??? ??? ??? ??? ?#endif
?? ??? ??? ??? ??? ??? ??? ?};
?? ??? ??? ??? ??? ??? ??? ?=>struct uart_state *state = tty->driver_data;
?? ??? ??? ??? ??? ??? ??? ?struct uart_port *port;
?? ??? ??? ??? ??? ??? ??? ?struct circ_buf *circ;
?? ??? ??? ??? ??? ??? ??? ?unsigned long flags;
?? ??? ??? ??? ??? ??? ??? ?=>port = state->uart_port;
?? ??? ??? ??? ??? ??? ??? ?circ = &state->xmit;
?? ??? ??? ??? ??? ??? ??? ?=>while (1) {
?? ??? ??? ??? ??? ??? ??? ??? ?c = CIRC_SPACE_TO_END(circ->head, circ->tail, UART_XMIT_SIZE);
?? ??? ??? ??? ??? ??? ??? ??? ?if (count < c)
?? ??? ??? ??? ??? ??? ??? ??? ??? ?c = count;
?? ??? ??? ??? ??? ??? ??? ??? ?if (c <= 0)
?? ??? ??? ??? ??? ??? ??? ??? ??? ?break;
?? ??? ??? ??? ??? ??? ??? ??? ?memcpy(circ->buf + circ->head, buf, c);
?? ??? ??? ??? ??? ??? ??? ??? ?circ->head = (circ->head + c) & (UART_XMIT_SIZE - 1);
?? ??? ??? ??? ??? ??? ??? ??? ?buf += c;
?? ??? ??? ??? ??? ??? ??? ??? ?count -= c;
?? ??? ??? ??? ??? ??? ??? ??? ?ret += c;
?? ??? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ??? ?=>uart_start(tty);
?? ??? ??? ??? ??? ??? ??? ??? ?=>struct uart_state *state = tty->driver_data;
?? ??? ??? ??? ??? ??? ??? ??? ?struct uart_port *port = state->uart_port;
?? ??? ??? ??? ??? ??? ??? ??? ?=>__uart_start(tty);
?? ??? ??? ??? ??? ??? ??? ??? ??? ?=>struct uart_state *state = tty->driver_data;
?? ??? ??? ??? ??? ??? ??? ??? ??? ?struct uart_port *port = state->uart_port;
?? ??? ??? ??? ??? ??? ??? ??? ??? ?=>if (!uart_circ_empty(&state->xmit) && state->xmit.buf && !tty->stopped && !tty->hw_stopped)
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?port->ops->start_tx(port);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?^
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?||
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?v
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?serial8250_start_tx
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?struct uart_ops serial8250_pops = {
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.tx_empty?? ?= serial8250_tx_empty,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.set_mctrl?? ?= serial8250_set_mctrl,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.get_mctrl?? ?= serial8250_get_mctrl,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.stop_tx?? ?= serial8250_stop_tx,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.start_tx?? ?= serial8250_start_tx,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.stop_rx?? ?= serial8250_stop_rx,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.enable_ms?? ?= serial8250_enable_ms,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.break_ctl?? ?= serial8250_break_ctl,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.startup?? ?= serial8250_startup,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.shutdown?? ?= serial8250_shutdown,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.set_termios?? ?= serial8250_set_termios,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.set_ldisc?? ?= serial8250_set_ldisc,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.pm?? ??? ?= serial8250_pm,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.type?? ??? ?= serial8250_type,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.release_port?? ?= serial8250_release_port,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.request_port?? ?= serial8250_request_port,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.config_port?? ?= serial8250_config_port,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.verify_port?? ?= serial8250_verify_port,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?#ifdef CONFIG_CONSOLE_POLL
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.poll_get_char = serial8250_get_poll_char,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?.poll_put_char = serial8250_put_poll_char,
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?#endif
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?};
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>struct uart_8250_port *up = (struct uart_8250_port *)port;
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>serial_out
?? ??? ??? ??? ??? ??? ?===>if (c < 0) {
?? ??? ??? ??? ??? ??? ??? ?retval = c;
?? ??? ??? ??? ??? ??? ??? ?goto break_out;
?? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ?if (!c)
?? ??? ??? ??? ??? ??? ??? ?break;
?? ??? ??? ??? ??? ??? ?b += c;
?? ??? ??? ??? ??? ??? ?nr -= c;
?? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ?==>if (!nr)
?? ??? ??? ??? ??? ??? ?break;
?? ??? ??? ??? ??? ?if (file->f_flags & O_NONBLOCK) {
?? ??? ??? ??? ??? ??? ?retval = -EAGAIN;
?? ??? ??? ??? ??? ??? ?break;
?? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ?schedule();
?? ??? ??? ??? ?=>__set_current_state(TASK_RUNNING);
?? ??? ??? ??? ?remove_wait_queue(&tty->write_wait, &wait);
?? ??? ??? ??? ?if (b - buf != nr && tty->fasync)
?? ??? ??? ??? ??? ?set_bit(TTY_DO_WRITE_WAKEUP, &tty->flags);
?? ??? ??? ??? ?return (b - buf) ? b - buf : retval;
?? ?=>tty_ldisc_deref(ld);
?? ?
?? ?

?? ??? ??? ??? ?

irqreturn_t serial8250_interrupt(int irq, void *dev_id)
?? ?=>struct irq_info *i = dev_id;
?? ?struct list_head *l, *end = NULL;
?? ?=>l = i->head;
?? ?=>do {
?? ??? ?struct uart_8250_port *up = list_entry(l, struct uart_8250_port, list);
?? ??? ?unsigned intiir = serial_in(up, UART_IIR);
?? ??? ?if (!(iir & UART_IIR_NO_INT)) {
?? ??? ??? ?serial8250_handle_port(up);
?? ??? ??? ??? ?=>status = serial_inp(up, UART_LSR);
?? ??? ??? ??? ?=>if (status & (UART_LSR_DR | UART_LSR_BI))
?? ??? ??? ??? ??? ?receive_chars(up, &status);
?? ??? ??? ??? ??? ??? ?=>struct tty_struct *tty = up->port.state->port.tty;
?? ??? ??? ??? ??? ??? ?unsigned char ch, lsr = *status;
?? ??? ??? ??? ??? ??? ?int max_count = 256;
?? ??? ??? ??? ??? ??? ?=>do {
?? ??? ??? ??? ??? ??? ??? ?if (likely(lsr & UART_LSR_DR))
?? ??? ??? ??? ??? ??? ??? ??? ?ch = serial_inp(up, UART_RX);
?? ??? ??? ??? ??? ??? ??? ?uart_insert_char(&up->port, lsr, UART_LSR_OE, ch, flag);
?? ??? ??? ??? ??? ??? ??? ??? ?=>struct tty_struct *tty = port->state->port.tty;
?? ??? ??? ??? ??? ??? ??? ??? ?=>if ((status & port->ignore_status_mask & ~overrun) == 0)
?? ??? ??? ??? ??? ??? ??? ??? ??? ?tty_insert_flip_char(tty, ch, flag);
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>struct tty_buffer *tb = tty->buf.tail;
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?if (tb && tb->used < tb->size) {
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?tb->flag_buf_ptr[tb->used] = flag;
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?tb->char_buf_ptr[tb->used++] = ch;
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?return 1;
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?}
?? ??? ??? ??? ??? ??? ??? ??? ??? ??? ?=>return tty_insert_flip_string_flags(tty, &ch, &flag, 1);
?? ??? ??? ??? ??? ??? ?} while ((lsr & (UART_LSR_DR | UART_LSR_BI)) && (max_count-- > 0));
?? ??? ??? ??? ??? ??? ?=>tty_flip_buffer_push(tty);
?? ??? ??? ??? ??? ??? ?=>*status = lsr;
?? ??? ??? ??? ?=>check_modem_status(up);
?? ??? ??? ??? ?=>if (status & UART_LSR_THRE)
?? ??? ??? ??? ??? ?transmit_chars(up);
?? ??? ??? ?handled = 1;

?? ??? ??? ?end = NULL;
?? ??? ?}
?? ??? ?} while (l != end);
?? ?=>return IRQ_RETVAL(handled);
?? ??? ?
?? ?
?? ?
?? ?
linux下串口(serial)和串口驅動
https://blog.csdn.net/wangzhen209/article/details/76685756

linux 串口驅動詳細分析
https://blog.csdn.net/dai_xiangjun/article/details/41241881

tty初探—uart驅動框架分析
https://blog.csdn.net/lizuobin2/article/details/51773305

serival(串口驅動)分析
https://wenku.baidu.com/view/49bc74c3b7360b4c2f3f6420.html

LINUX 日志級別(LOGLEVEL)詳解
http://smilejay.com/2011/12/linux_loglevel/

Linux中tty框架與uart框架之間的調用關系剖析
http://blog.chinaunix.net/uid-29025972-id-4738659.html

linux關于串口的配置和多線程收發
http://www.360doc.com/content/17/0831/23/44391309_683701497.shtml

Linux下的串口編程及非阻塞模式
https://www.cnblogs.com/ynxf/p/6105072.html

Linux open系統調用流程淺析
https://www.jianshu.com/p/f3f5a33f2c59

淺析linux中open系統調用
http://www.360doc.com/content/12/0507/15/9171956_209262761.shtml

ttyUSB串口設備節點生成過程
https://blog.csdn.net/mingtianwoyueni/article/details/63709861

Linux串口驅動分析read
https://blog.csdn.net/longwang155069/article/details/42776059

tty初探—uart驅動框架分析
https://blog.csdn.net/lizuobin2/article/details/51773305

串口驅動分析
https://blog.csdn.net/weed_hz/article/details/8946391

linux設備驅動之8250串口驅動
https://blog.csdn.net/zjy900507/article/details/78678783

總結

以上是生活随笔為你收集整理的慢慢欣赏linux之串口驱动代码分析 - 基于powerpc 2.6.x版本的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

精品视频免费看 | 香蕉视频导航 | 亚洲免费视频观看 | 日本色小说视频 | 蜜臀久久99精品久久久无需会员 | 国产字幕av | 欧美极品一区二区三区 | 日韩 在线观看 | 亚州免费视频 | 亚洲女欲精品久久久久久久18 | 久草网视频 | 亚洲精品xxxx| 国产成人精品一区一区一区 | 91黄视频在线观看 | 日韩精品久久中文字幕 | 免费三级av | 人人爽人人片 | 在线观看黄av | 国产无套精品久久久久久 | 久久精品系列 | 色香蕉视频 | 久久精品人| 国产精品久久久久久久久久久免费看 | 亚欧日韩av | 久久夜靖品| 久久久亚洲麻豆日韩精品一区三区 | 精品久久网站 | 欧美日韩高清在线 | 国产不卡在线观看视频 | 中文国产成人精品久久一 | 成人av网站在线播放 | 男女拍拍免费视频 | 中文字幕字幕中文 | 一本一本久久a久久精品综合小说 | 亚洲综合成人专区片 | 国产剧情av在线播放 | 99久精品| 欧美视频在线二区 | 九九热有精品 | 91尤物国产尤物福利在线播放 | 黄污视频网站 | 久久国产影院 | av免费在线看网站 | 久99久视频 | 国产精品免费在线 | 久久精品国产一区二区 | 久久专区| 天天操天天舔天天干 | 国产18精品乱码免费看 | 江苏妇搡bbbb搡bbbb | 日韩av一区二区三区 | 一区二区三区 中文字幕 | 成人一级在线观看 | 国产蜜臀av | 久久国产免费看 | 日本xxxx裸体xxxx17 | 91av在线免费看 | 九九久久国产 | 国产高清在线a视频大全 | 在线看片中文字幕 | www.av在线播放 | 草久在线观看视频 | 免费高清在线观看成人 | 波多野结衣电影一区 | 在线观看91视频 | 久久久久久久久毛片 | 免费黄在线观看 | 91久久丝袜国产露脸动漫 | 四虎成人精品在永久免费 | 婷婷丁香激情网 | 日韩精品亚洲专区在线观看 | 99精品免费久久久久久久久 | av再线观看 | 国产精品9999久久久久仙踪林 | a视频在线播放 | 日韩视频免费在线观看 | 正在播放国产一区二区 | 亚洲一级久久 | 国产精品网址在线观看 | 国产精品青草综合久久久久99 | 欧美激情综合色 | 成 人 a v天堂| 免费色视频| 婷婷深爱网 | 免费三级黄色 | 激情视频一区二区 | 在线日韩av| 日韩av在线一区二区 | 日韩国产精品久久久久久亚洲 | 色婷婷av一区二 | 久久这里只有精品9 | 丁香婷婷色月天 | av不卡免费在线观看 | 日批视频| 91人人澡人人爽人人精品 | 激情视频在线高清看 | 91系列在线观看 | 欧美性脚交| 久久不卡国产精品一区二区 | 国产一级电影网 | 人人爽久久涩噜噜噜网站 | 国产一级片免费播放 | 欧美激情另类文学 | 久草电影免费在线观看 | 久草在线视频中文 | 香蕉影视在线观看 | 久久久久久久久影视 | 欧美福利视频一区 | 91久久精品日日躁夜夜躁国产 | 久久婷亚洲五月一区天天躁 | 国产乱对白刺激视频在线观看女王 | www国产亚洲精品久久网站 | 91香蕉视频好色先生 | 成人av中文字幕 | 日韩中字在线 | 亚洲综合在线观看视频 | 午夜色大片在线观看 | 我爱av激情网 | 午夜精品久久久久久99热明星 | 国产又粗又猛又黄视频 | 特级黄色一级 | 久久久久免费精品视频 | 中文字幕 婷婷 | 天天舔夜夜操 | 午夜精品久久久久久久久久 | 国产精品中文字幕av | 久久99在线 | 国产一线二线三线性视频 | 不卡的av中文字幕 | 午夜免费视频网站 | 久爱精品在线 | 欧美日韩在线视频观看 | 日韩 精品 一区 国产 麻豆 | 久久精品免费电影 | 国产成人精品一区二区在线观看 | 国产精品永久久久久久久www | 人人网av| 五月婷婷激情网 | 99精品在线免费 | 狠狠干天天色 | 激情电影影院 | 国产只有精品 | 欧美日韩观看 | 中文字幕亚洲国产 | 亚洲国产精彩中文乱码av | 久久精品美女视频 | 97成人在线观看视频 | 五月婷在线视频 | 伊人网av | 日韩欧美在线免费 | 久草爱视频 | 精品欧美日韩 | 99久久精品免费看国产免费软件 | 美女免费视频观看网站 | 麻豆免费在线视频 | 久久久久久久久久久免费视频 | 午夜精品久久久久久 | 奇米网在线观看 | 精品国产精品国产偷麻豆 | 久久久久国产精品免费免费搜索 | 69国产盗摄一区二区三区五区 | 久久99久久99 | 久久久久久久久久久久影院 | 中文字幕中文字幕在线中文字幕三区 | 日本爽妇网 | 韩国av免费观看 | 久久久久久久久久久久av | 午夜精品电影一区二区在线 | 最近中文字幕mv免费高清在线 | 奇米影视8888在线观看大全免费 | 久久精品看| h动漫中文字幕 | 亚洲精品视频免费看 | 亚洲国产精品电影 | 黄色大片视频网站 | 欧美极度另类性三渗透 | 97人人添人澡人人爽超碰动图 | 久久久免费在线观看 | 久久综合五月天 | 99久久久久免费精品国产 | 亚洲一区精品二人人爽久久 | 久久a v视频 | 成人午夜网 | 精品久久久久久久久久 | 久草在线观 | 国产在线精品播放 | 国产精品久久一 | 久久精品视频网 | 91国内在线视频 | 人操人| 2023天天干| 99精品视频在线免费观看 | 黄色大片日本免费大片 | 国外调教视频网站 | 91麻豆产精品久久久久久 | 中文字幕丰满人伦在线 | 五月婷婷在线观看视频 | 96久久欧美麻豆网站 | 欧美 日韩 成人 | 欧美性久久久久久 | 欧美综合在线视频 | 欧美日韩国产二区 | 国产一级免费观看 | 中午字幕在线观看 | 亚洲电影成人 | 西西www4444大胆在线 | 国产精品99精品久久免费 | 免费看成人片 | 久久久免费av| 国产一级免费在线观看 | 亚洲aaa级| 国产精品久久久久一区二区国产 | 欧美国产亚洲精品久久久8v | 国产精品久久久久久久久久了 | 国产精品久久久久aaaa九色 | 在线观看国产成人av片 | 国产精品成久久久久三级 | 视频国产在线观看18 | 亚洲国产黄色片 | 国产网站av | 国产丝袜网站 | 天天爽夜夜爽人人爽一区二区 | 国产精品女同一区二区三区久久夜 | 亚洲乱码精品久久久 | 97视频人人澡人人爽 | 麻豆一二 | 中文字幕av最新更新 | 国产伦精品一区二区三区四区视频 | 999久久久欧美日韩黑人 | 久久久久福利视频 | 欧美精品久久 | 国产精品久久网 | 国产精品欧美久久久久久 | а天堂中文最新一区二区三区 | 成人a免费看 | 日日草av| 日韩电影精品 | 国产美女免费 | 五月av在线 | 免费a一级 | 在线视频1卡二卡三卡 | 国产精品剧情 | 黄色av免费看 | 免费观看一级成人毛片 | 久久欧美在线电影 | av黄网站 | 综合伊人av | 成人网444ppp | 精品影院一区二区久久久 | 欧美精品一区二区在线观看 | av黄色在线播放 | 欧美一区二区在线免费看 | 国产91丝袜在线播放动漫 | 国产亚洲精品成人av久久ww | 国产免费av一区二区三区 | 99久高清在线观看视频99精品热在线观看视频 | 五月婷婷综合久久 | 日韩免费一级a毛片在线播放一级 | 天天操天操| 中文字幕日本特黄aa毛片 | 少妇精品久久久一区二区免费 | 午夜12点| 黄色aa久久 | 中文字幕在线观看免费高清完整版 | 国产精品九九久久久久久久 | 国产一级特黄毛片在线毛片 | 欧美一级片| 国产精品12345 | 日韩超碰| 成人免费视频播放 | 午夜婷婷在线观看 | 亚洲精品动漫成人3d无尽在线 | 五月花丁香婷婷 | 欧美一二在线 | 天堂av在线中文在线 | 天天干天天在线 | 亚洲人久久久 | 久久综合之合合综合久久 | 国产中文字幕一区 | 免费国产亚洲视频 | 成人免费在线观看入口 | 久久视频网址 | 高清不卡毛片 | 国产色秀视频 | 日韩高清不卡在线 | 日韩毛片一区 | 天天操夜夜摸 | 亚洲午夜久久久久久久久电影网 | 天天艹天天干天天 | 日韩av电影免费观看 | 天天干视频在线 | 99热高清| 最近的中文字幕大全免费版 | 337p日本大胆噜噜噜噜 | 国产91aaa | 欧美精品久久久久久久久免 | 中文日韩在线 | 国产又粗又猛又色又黄网站 | 日韩在线无 | 国产精品久久久久久一区二区 | 奇米网8888 | 免费下载高清毛片 | 日韩在线电影 | 欧美日韩xxx | 亚洲一级片免费观看 | 国产视频在线观看一区 | 国产色视频123区 | 欧美日韩破处 | 岛国一区在线 | 国产精品资源网 | 91精品久久久久久久99蜜桃 | 国产午夜一区二区 | 久久久久久在线观看 | 国产精品成人一区二区三区吃奶 | 美女在线国产 | 99精品视频在线 | 亚洲欧美综合精品久久成人 | 亚洲综合视频在线 | 黄色影院在线免费观看 | 色狠狠综合天天综合综合 | 国产视频在| 亚洲春色综合另类校园电影 | 91麻豆精品| 亚洲免费激情 | 久久精品视频在线观看免费 | 视频91在线| 97在线观 | 国产美女精品视频 | 在线观看深夜福利 | 亚洲成人资源 | 久久精品亚洲国产 | av一级在线 | 亚洲精品国产精品国 | 91久久黄色 | 中文字幕资源网在线观看 | 久久免费视频国产 | 91看片在线| 国产精品久久久久久婷婷天堂 | 亚洲最大av | www.久热| 久久成人午夜 | 中字幕视频在线永久在线观看免费 | 亚洲欧美日韩精品久久奇米一区 | 国产一级一片免费播放放a 一区二区三区国产欧美 | 成 人 黄 色 视频免费播放 | 色吊丝在线永久观看最新版本 | 日本精品视频免费观看 | 国产精品片| 日p在线观看 | 色偷偷男人的天堂av | 中文在线天堂资源 | 中文字幕在线观看你懂的 | 91 中文字幕| 日韩av午夜 | 欧美日韩xxx | 免费观看黄 | 在线亚洲人成电影网站色www | 中文永久字幕 | 91亚洲精品久久久久图片蜜桃 | 国产视频在线观看一区 | 精品欧美一区二区在线观看 | 久久人人97超碰精品888 | 亚洲精品国产品国语在线 | 亚洲视频1 | 国产一级特黄毛片在线毛片 | 最近中文字幕免费视频 | 国产精品 国内视频 | 开心色停停| 99久久这里有精品 | 欧美综合在线观看 | 成人免费 在线播放 | 亚洲国产精品一区二区尤物区 | 97国产| 亚洲人在线 | 免费观看久久 | 特级毛片在线观看 | 国产网红在线观看 | 国产精品地址 | 免费三及片 | 久久精品日产第一区二区三区乱码 | 99久久精品无码一区二区毛片 | 在线不卡的av| 最近日本mv字幕免费观看 | 亚洲国产成人在线 | 免费高清在线观看电视网站 | 国产18精品乱码免费看 | 国产精品一区二区在线免费观看 | 丁香六月婷婷开心 | 国内精品视频免费 | 日本精品久久久久久 | 草樱av | 国产a视频免费观看 | 久草在线官网 | 午夜精品福利一区二区三区蜜桃 | a级国产乱理论片在线观看 伊人宗合网 | 免费日韩 精品中文字幕视频在线 | 91久久精品日日躁夜夜躁国产 | 天天天天干 | 日韩在线不卡av | 亚洲视频 在线观看 | 在线免费观看视频a | 亚洲国产日韩欧美在线 | 97中文字幕 | 午夜婷婷网 | 四虎4hu永久免费 | 国产精品国产三级国产aⅴ9色 | 久久99国产精品久久99 | 国产成人精品av久久 | 精品久久91 | 国产91在线播放 | 国产黄色精品在线观看 | 又黄又爽免费视频 | 精品久久一 | 欧美伦理一区 | 日韩精品久久一区二区 | 亚洲激情一区二区三区 | 久久国产精品视频免费看 | 五月天av在线 | 最新日韩视频 | 国产精品黄色影片导航在线观看 | 久热精品国产 | 天干啦夜天干天干在线线 | 国产精华国产精品 | 久久中文字幕导航 | 超碰免费在线公开 | 国产成人一区二区精品非洲 | 国产精品毛片完整版 | 国模精品在线 | 国产在线 一区二区三区 | 亚洲综合少妇 | 免费av网站观看 | 在线成人免费 | 韩国三级av在线 | 精品无人国产偷自产在线 | 激情欧美在线观看 | 一级成人在线 | 少妇自拍av | 午夜久久福利影院 | 亚洲精品成人 | 最近日本mv字幕免费观看 | 黄污网| 免费在线观看视频a | 99久久免费看 | 综合天天| 亚洲综合色av | www.久久久.com | 精品国产亚洲日本 | 国产精品美女久久久免费 | 亚洲粉嫩av| 深夜视频久久 | 国产精品免费久久久久久 | 三级在线国产 | 久久免费视频精品 | 久99久精品 | www.夜夜操| 国产精品久久艹 | 天天色天天综合 | 青草视频在线看 | 在线精品亚洲一区二区 | 成人免费在线观看入口 | 日韩亚洲在线视频 | 久久精品激情 | 97人人视频| 97超碰人人澡 | 天天操天天插 | 日本在线观看黄色 | 国产麻豆剧传媒免费观看 | 国产伦理久久精品久久久久_ | 99国产在线观看 | 又紧又大又爽精品一区二区 | 久久视频国产精品免费视频在线 | 免费合欢视频成人app | 热久久视久久精品18亚洲精品 | 日日狠狠| www.夜夜爽 | 97超碰人人网| 91精品导航 | 91av精品 | av一级在线| 六月婷婷网 | 亚洲免费观看视频 | 97香蕉久久国产在线观看 | 97超碰资源 | 国产在线国产 | 一区二区视频在线免费观看 | 日韩特黄av | 欧美xxxx性xxxxx高清 | 欧美少妇xxx| 日韩在线首页 | 伊人久在线 | 久草视频免费在线观看 | 国产精品高潮在线观看 | 国产精品伦一区二区三区视频 | 亚洲精品动漫成人3d无尽在线 | 日韩网站在线免费观看 | 久久久国产精品视频 | 91大神精品视频 | 少妇自拍av | 色综合色综合色综合 | 欧美久久久久久久久久久久 | 亚洲精品在线播放视频 | 国产一区二区手机在线观看 | 91自拍视频在线观看 | 中文字幕av专区 | 国产精品视频久久久 | 亚洲国产无 | www.888.av| 亚洲精品国产综合99久久夜夜嗨 | 国产精品99久久免费黑人 | 日批在线看| 亚洲天堂网在线视频 | 精品色综合 | 国产在线不卡精品 | 欧美日韩在线观看一区二区三区 | 欧美韩日精品 | 亚洲狠狠丁香婷婷综合久久久 | 国产在线中文字幕 | 国产精品一区二区在线观看免费 | 91精品一区二区三区蜜桃 | 婷婷激情5月天 | 国产99久久久精品视频 | 久久精品国产精品亚洲 | 最新日韩在线观看 | 天天草天天干天天 | 香蕉在线视频观看 | 天天夜操 | 日韩va欧美va亚洲va久久 | 91免费在线视频 | 91麻豆国产福利在线观看 | 91中文字幕在线观看 | 日韩免费在线视频观看 | 国产主播99| 日韩电影在线看 | 久久国产99 | 91一区啪爱嗯打偷拍欧美 | 97在线超碰 | 亚洲一区动漫 | 91av超碰| adc在线观看 | 欧美福利网址 | 日p在线观看 | 偷拍久久久 | 精品一区二区免费视频 | 国产免费专区 | 91视频麻豆视频 | 97激情影院 | 日批网站免费观看 | 日韩成人不卡 | 免费看黄的 | 欧美日韩1区 | 婷婷色综 | 99视频在线 | 久草视频在线免费 | 综合婷婷 | 成人在线播放免费观看 | 黄污在线看 | 亚洲成人av片 | 日韩网站一区二区 | 久久久精品综合 | av综合站| 91丨九色丨蝌蚪丨对白 | 色诱亚洲精品久久久久久 | 91av在线看| 日韩免费在线看 | 久久久国产精品一区二区中文 | 久久精品一区二区三区中文字幕 | 中文av不卡 | 亚洲精品免费在线 | 99理论片 | 福利二区视频 | 依人成人综合网 | 欧美日韩亚洲一 | 天天操狠狠操网站 | 91香蕉视频黄色 | 国产成年免费视频 | 人人干人人添 | 婷婷六月中文字幕 | 99re6热在线精品视频 | 中国老女人日b | 国产精国产精品 | 丝袜足交在线 | 尤物97国产精品久久精品国产 | 日韩精品中文字幕有码 | 精品亚洲成人 | 欧美久久久久久久久中文字幕 | 蜜桃视频在线视频 | 久久一级电影 | 国产成人精品久久二区二区 | 欧美激情精品久久久 | 99色在线观看 | 四虎在线观看精品视频 | 91网页版免费观看 | 青春草视频在线播放 | 国产亚洲日 | 999热视频| 东方av在线免费观看 | 国产一区二区三区高清播放 | 91亚洲在线 | 一区二区三区免费在线观看 | 中文av日韩| 九九久久久久99精品 | 97网站| 色播五月婷婷 | 涩涩资源网| 久草在线视频中文 | 国产午夜三级一区二区三桃花影视 | 亚洲不卡在线 | 久久久视频在线 | 人人dvd | 亚洲免费在线观看视频 | 成人黄色片免费 | 8x成人在线 | 欧美疯狂性受xxxxx另类 | 911国产| 777xxx欧美 | 成人免费视频网站在线观看 | 天天伊人狠狠 | www.亚洲激情.com | www视频在线观看 | 91九色视频在线观看 | av中文在线观看 | 久久免费黄色网址 | 九色精品在线 | 久久精彩| 精品久久久久国产免费第一页 | 91中文字幕在线播放 | 色婷婷五| 中文字幕在线播放一区二区 | av黄色一级片 | www.久久精品视频 | 97精产国品一二三产区在线 | 探花在线观看 | 五月天电影免费在线观看一区 | 欧美激情第一区 | 久久久免费精品视频 | 欧美一区二视频在线免费观看 | 久久看毛片 | 久草五月 | 国产主播大尺度精品福利免费 | 成人小视频免费在线观看 | 成人免费大片黄在线播放 | 麻花豆传媒mv在线观看网站 | 国产精品一区在线观看 | 三级av在线 | 欧美性色黄大片在线观看 | 91传媒激情理伦片 | 国产69精品久久久久久 | 国产精品九九九九九 | 91视频这里只有精品 | 字幕网资源站中文字幕 | 免费精品国产va自在自线 | 91久久久久久国产精品 | 国产成人一区二区三区在线观看 | 欧美日韩裸体免费视频 | 免费亚洲精品视频 | 91最新地址永久入口 | 91传媒免费在线观看 | av成人在线观看 | 亚洲污视频 | 亚洲精品欧美精品 | www免费网站在线观看 | 日韩一级成人av | 免费高清av在线看 | 人人干网站 | 欧美aa在线 | 日韩电影一区二区在线观看 | 日韩三级.com | www.黄色片网站 | 日韩一区二区在线免费观看 | 免费看黄色大全 | 久久久精品 | 午夜男人影院 | 午夜久久视频 | 看v片| 久久99亚洲精品 | 五月婷婷丁香 | 国产很黄很色的视频 | 2019av在线视频 | 一区二区三区在线观看中文字幕 | 中文字幕一区二区在线播放 | 麻豆一区二区 | 黄色视屏在线免费观看 | 久久综合九色综合久99 | 国产精品久久久久亚洲影视 | 日本成人免费在线观看 | 久草影视在线 | 国产精品亚洲片在线播放 | 天天射天天操天天 | 国产不卡在线 | 亚洲精品高清视频 | 久久99精品久久久久久秒播蜜臀 | www黄色软件 | 色久综合| 国产成人在线免费观看 | 亚洲国产精品电影 | 天天爱天天操天天射 | www五月天com| 久久精品欧美一区 | 丰满少妇在线观看网站 | 亚洲精品福利在线 | 韩国一区二区av | 欧美日韩一区二区久久 | 国产不卡在线看 | 99久久www | 一区二区中文字幕在线 | www.夜色321.com| 色九九在线 | 精品国产一区二区三区免费 | 99久久999久久久精玫瑰 | 成年人在线免费看视频 | 亚洲 欧美日韩 国产 中文 | 欧美精彩视频在线观看 | 亚洲电影影音先锋 | 欧美激情xxxx性bbbb | 337p日本大胆噜噜噜噜 | 伊人中文字幕在线 | 人人插人人 | 亚洲天堂网站 | 五月开心综合 | 区一区二区三在线观看 | 91精品国产一区二区在线观看 | 国产三级视频 | 欧美极度另类 | 片网站| 久久99最新地址 | 日韩三级视频 | 中文亚洲欧美日韩 | 天天插天天操天天干 | 成人在线观看免费视频 | 视频在线观看亚洲 | 午夜精品久久久久久久99 | 国产在线一线 | 日本激情视频中文字幕 | 国产视频一区二区在线播放 | 五月天久久久久 | 97超碰免费 | 亚洲精品欧洲精品 | 久久精品国产第一区二区三区 | 天天躁天天狠天天透 | 日韩中文字幕国产 | 国产高清视频在线 | 青春草视频在线播放 | 日韩中字在线 | 免费高清在线观看成人 | 日韩系列| 欧美成人在线网站 | 国产最新91 | 西西4444www大胆无视频 | 久久99国产精品 | 99在线观看免费视频精品观看 | 激情av网址 | 欧美激情第一区 | 五月天婷婷免费视频 | 91传媒视频在线观看 | 久久午夜网 | 欧美国产日韩在线视频 | 九九在线高清精品视频 | 中文一二区 | 开心激情五月婷婷 | 天天婷婷| 狠狠操狠狠操 | 五月激情在线 | 色综合天天色综合 | 中文字幕在线视频网站 | 亚洲视频久久 | 久久精品79国产精品 | 国产福利不卡视频 | 黄色精品一区二区 | 黄色a在线 | www.亚洲视频 | 黄色在线视频网址 | 黄色三级免费片 | 国产美女视频 | 国产精品一区二 | 日韩va在线观看 | 免费在线观看a v | 久久精品日本啪啪涩涩 | 久久不卡免费视频 | 亚洲欧美日本一区二区三区 | 五月激情视频 | 人人射av| 在线免费观看不卡av | 天天色天天草天天射 | 91亚洲精品久久久久图片蜜桃 | 99久久久久| 免费av大片 | 久久99精品国产一区二区三区 | av天天干| 美女网站黄在线观看 | 国产精品成人国产乱 | 麻豆视频国产精品 | 三级黄色在线观看 | 欧美另类xxx | 在线黄色毛片 | 色偷偷人人澡久久超碰69 | 91最新网址在线观看 | 99理论片 | freejavvideo日本免费 | 国产精品久久久久永久免费看 | 91完整视频| 免费观看完整版无人区 | 91视频a | 色av婷婷| 日韩视频免费在线观看 | 97超级碰碰 | 久久任你操 | 成年人在线免费看视频 | 日本激情中文字幕 | 日韩精品一区二区三区中文字幕 | 久久精品视频中文字幕 | 综合色久 | 日韩精品无码一区二区三区 | 久久99国产综合精品 | 中文字幕a在线 | 久久久精品国产免费观看一区二区 | 亚洲涩涩色 | 99爱精品在线 | 久久综合色天天久久综合图片 | 日日摸日日 | 日韩免费播放 | 欧美一二三区播放 | 国内免费久久久久久久久久久 | 成人在线免费看视频 | 又污又黄的网站 | 激情网色| 国产亚洲高清视频 | 欧美精品在线观看免费 | 在线国产日本 | 欧美成人在线免费观看 | 国产伦精品一区二区三区无广告 | 超碰97人 | 免费看国产视频 | av中文字幕亚洲 | 网站在线观看日韩 | www.97色.com | 五月激情六月丁香 | 狠狠网站 | 日本精品xxxx | 免费午夜在线视频 | 狠狠狠狠狠狠狠 | 国产999免费视频 | 免费看黄在线观看 | 欧美黄色成人 | 五月开心婷婷网 | 超碰在线9| 在线99| 最近中文字幕大全 | 狠狠狠色丁香综合久久天下网 | 五月天综合网站 | 99久久99久久精品国产片 | 亚洲成人中文在线 | 五月综合在线观看 | 在线看污网站 | 探花视频在线版播放免费观看 | 爱射综合 | 日本资源中文字幕在线 | 欧美在线一二 | 亚洲成人网av | 久久 一区 | 国产不卡视频 | 欧美日韩性 | 欧美激情精品久久久久 | 久久久久久国产一区二区三区 | 免费看污污视频的网站 | 精品国产成人av | 手机av观看 | 开心激情五月网 | 日本中文字幕电影在线免费观看 | av在线播放不卡 | 精品久久1| 伊人激情综合 | 欧美日韩久久一区 | 99热在线观看 | 色婷婷六月| 激情在线网址 | 欧美在线日韩在线 | 日韩毛片在线一区二区毛片 | 婷婷丁香导航 | 新版资源中文在线观看 | 日韩视频中文字幕在线观看 | 免费观看mv大片高清 | 久久精品国产免费看久久精品 | 国产成人a亚洲精品 | 久久avav | 狠狠躁夜夜躁人人爽视频 | 欧美经典久久 | 国产精品一二三 | 99在线精品视频在线观看 | 99riav1国产精品视频 | 91网址在线观看 | 在线之家免费在线观看电影 | 国产精品麻豆果冻传媒在线播放 | 成人亚洲网 | 九九在线免费视频 | 国产精彩在线视频 | 2017狠狠干| 亚洲理论在线 | 国产视频 亚洲精品 | 亚洲欧洲国产日韩精品 | 在线观看视频你懂的 | 久久97精品| 日韩h在线观看 | 二区三区精品 | 日本久久久精品视频 | 午夜av免费看| 中文字幕在线观看的网站 | www.av在线播放 | 激情久久一区二区三区 | 亚洲视频精品在线 | 国产精品免费小视频 | 国外调教视频网站 | 成人资源在线播放 | 国产亚洲无 | 亚洲精品网站在线 | 精品黄色片 | 国产精品一区二区三区四区在线观看 | 天天伊人网 | 97国产精品亚洲精品 | 日韩专区中文字幕 | 在线免费观看国产黄色 | 成人国产网站 | 黄色的视频网站 | 国产精品资源 | 亚州精品视频 | 丁香六月中文字幕 | 天天操天天射天天操 | 96香蕉视频 | 国产精品自产拍在线观看 | 久久av中文字幕片 | 免费看的国产视频网站 | 久草在线最新视频 | 中文字幕中文字幕在线中文字幕三区 | 成人一区二区在线 | 日韩网站一区二区 | 亚洲区另类春色综合小说校园片 | 香蕉手机在线 | 在线视频日韩欧美 | 蜜臀久久99静品久久久久久 | 九色精品 | 欧美巨乳波霸 | 日韩高清久久 | 日韩网站免费观看 | 91成人网在线观看 | 日本精品一区二区三区在线观看 | 成人三级网址 | 99热9 | 国产在线第三页 | 精品国产自在精品国产精野外直播 | 又爽又黄又无遮挡网站动态图 | 亚洲精品97| 日韩欧美高清一区二区 | 精品电影一区二区 | 精品产品国产在线不卡 | av在线在线| 亚洲免费不卡 | 中国精品少妇 | 欧美久久久一区二区三区 | www.久艹 | 成人影片免费 | 91av视频 | 欧美日韩调教 | 在线黄av| 婷婷五综合 | 欧美看片| 深夜免费网站 | 天天操天天干天天爱 | 婷五月天激情 | 国产精品6 | 99爱视频| 97国产情侣爱久久免费观看 | 综合婷婷 | 午夜精品久久一牛影视 | 国产午夜三级一区二区三桃花影视 | 四虎在线免费观看视频 | 黄色精品在线看 | 天天玩天天干 | 国产成人综 | 四川妇女搡bbbb搡bbbb搡 | 蜜臀av性久久久久av蜜臀三区 | 麻豆国产在线视频 | 97av色| 婷婷亚洲综合 | 国产高清视频在线播放一区 | 久久99国产精品 | 日韩网站视频 | 日韩欧美国产精品 | 中国一区二区视频 | www久草| 丁香花在线视频观看免费 | 免费观看国产视频 | 97精品国产97久久久久久久久久久久 | 丁香婷婷激情五月 | av大片网站 | 久久短视频 | 欧美国产日韩久久 | 日本精品一区二区在线观看 | 黄色软件在线观看免费 | 久久首页 |