日韩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版本的全部內容,希望文章能夠幫你解決所遇到的問題。

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

国产最新91 | 三上悠亚一区二区在线观看 | 亚洲成人软件 | 免费成人av在线看 | 精品国产一区二区三区久久久 | 日韩va亚洲va欧美va久久 | 91精品人成在线观看 | av在线电影免费观看 | 久久精品中文字幕一区二区三区 | 在线观看中文av | 天天爽天天搞 | 最近最新中文字幕视频 | 五月天综合色 | 中文在线字幕观看电影 | 欧美一区二区三区在线观看 | 美女视频黄免费 | 国产成人精品免高潮在线观看 | 999久久a精品合区久久久 | 久久视 | 国产在线一卡 | 激情视频综合网 | 亚洲色视频 | 免费看搞黄视频网站 | 精品在线观看国产 | 国产91在线免费视频 | 午夜影院先 | 亚洲女裸体 | 91香蕉视频在线下载 | 国产亚洲免费的视频看 | 精品一区 在线 | 国产精品mv在线观看 | 国产精品一区二区免费在线观看 | 在线观看亚洲视频 | 亚洲第一区精品 | 91超在线| 韩日电影在线观看 | 久久在线免费视频 | 天天草夜夜 | 丁香综合 | 一级黄色免费网站 | 黄色小说视频在线 | 日韩经典一区二区三区 | 欧美成人黄色 | 香蕉视频网址 | 国产福利精品一区二区 | 高清中文字幕 | 精品在线观看国产 | 亚洲永久精品国产 | 黄色激情网址 | 五月天久久久 | 99久久婷婷国产 | 日韩久久精品一区二区 | 操操操夜夜操 | 日韩国产欧美视频 | 久久免费av | 91成人亚洲 | 欧美日韩二区三区 | 午夜精品久久 | 久久久在线观看 | 日本中文字幕系列 | 13日本xxxxxⅹxxx20 | 成人午夜电影久久影院 | 黄色小说视频网站 | www.黄色小说.com | 香蕉久久久久久av成人 | 99在线免费视频 | 午夜视频亚洲 | 91九色porn在线资源 | 精品国产伦一区二区三区观看体验 | 操夜夜操 | 欧美性大战久久久久 | 久久男人中文字幕资源站 | 免费视频黄 | va视频在线观看 | 午夜123| 97精产国品一二三产区在线 | 精品女同一区二区三区在线观看 | 91九色蝌蚪视频在线 | 天天曰天天 | 国产黑丝一区二区三区 | 欧美大片在线看免费观看 | 国产在线观看高清视频 | 国产黄色免费看 | 在线看国产一区 | 久久久久激情 | 欧美影片 | 国产手机av在线 | 综合久久久久久久 | 中文字幕在线观看第一区 | 中文字幕国产精品 | 久草在线这里只有精品 | 亚洲婷婷丁香 | 国产精品青草综合久久久久99 | 久久久久国产精品免费网站 | av先锋中文字幕 | 一区二区三区日韩视频在线观看 | 国产中文字幕视频 | 久久国产精品视频免费看 | 国产日韩精品在线 | 精品久久久久久久久久久院品网 | 操久在线 | 久久超碰在线 | 欧美在线观看视频一区二区三区 | 97精品在线视频 | 国产精品中文在线 | 成人小视频免费在线观看 | 日韩欧美xxxx| 日韩色区 | 九九热在线精品 | 日韩一区在线免费观看 | 国产小视频在线 | 久久久久免费精品国产小说色大师 | 日日操网 | 国产精品久久久久久久久久久久冷 | 91麻豆精品国产91久久久无限制版 | 亚洲高清久久久 | 久青草电影 | 黄色亚洲精品 | 色婷婷 亚洲 | 国产一级二级三级在线观看 | japanesefreesex中国少妇 | 天天天射 | 黄色av电影一级片 | 久久国产精品成人免费浪潮 | 国产精品99久久久久久久久久久久 | 日韩精品一区在线观看 | 九九热.com| 国产九九精品 | 又黄又爽的免费高潮视频 | 亚洲一区二区三区精品在线观看 | 99热 精品在线| 日韩性片| 久久午夜影院 | 欧美极度另类性三渗透 | 日韩精品一区在线播放 | 黄色av高清| 国产一区二区三区四区大秀 | 日韩视频免费观看高清完整版在线 | 成人免费视频播放 | 99久久精品国产欧美主题曲 | 天天干天天干天天 | 91激情视频在线观看 | 国产xxxxx在线观看 | 国产精品黄色 | 国产一区高清在线观看 | av电影不卡在线 | 夜色成人网 | 中文字幕在线观看第二页 | 人人爽人人爽人人爽 | 97精品伊人 | 婷婷播播网 | 日日婷婷夜日日天干 | 午夜私人影院久久久久 | 日本h视频在线观看 | 丝袜美腿亚洲 | 欧美一二三区在线观看 | 婷婷精品国产一区二区三区日韩 | 热久在线 | 日本黄色一级电影 | 伊人看片 | 最近2019中文免费高清视频观看www99 | 欧美精品久久 | a√天堂中文在线 | 久久影院午夜论 | 久久久免费看视频 | 精品九九九 | 五月婷婷开心中文字幕 | 久久综合九色99 | 黄色亚洲精品 | 国产一级二级在线播放 | 亚洲综合网站在线观看 | av直接看| 五月激情五月激情 | 亚洲无人区小视频 | 波多野结衣亚洲一区二区 | 亚洲天堂网站视频 | 日韩欧美视频免费在线观看 | 一区二区三区四区不卡 | 美女性爽视频国产免费app | 成人午夜影院在线观看 | 探花视频在线观看+在线播放 | 成人一级 | 精品一区 在线 | 日韩欧美网址 | 毛片永久新网址首页 | 国产又粗又猛又黄视频 | 国产黄色免费在线观看 | www.色com | 欧美亚洲久久 | 国产伦理久久精品久久久久_ | av电影在线观看完整版一区二区 | 视频二区在线视频 | 中文字幕国内精品 | 97成人在线免费视频 | 国产999视频在线观看 | 欧美天天综合网 | 日韩精品在线免费观看 | 欧美福利在线播放 | a在线免费| 午夜精品电影一区二区在线 | 亚洲精品成人免费 | 欧美日韩国产精品一区 | 久久久久国产一区二区三区 | 97夜夜澡人人双人人人喊 | 免费在线观看日韩欧美 | 国产在线日本 | 欧美韩国日本在线 | 久免费 | 在线免费观看国产视频 | 在线视频区 | 国产一区在线视频 | 人人爽人人爽av | 成人精品一区二区三区中文字幕 | 在线观看免费av片 | 久久久久国产精品午夜一区 | 日韩视频专区 | 国产精品久久精品国产 | 欧美午夜精品久久久久 | 免费黄在线看 | 亚洲精品资源在线观看 | 9久久精品 | 国产 日韩 欧美 中文 在线播放 | 亚洲区精品 | 91色吧| 久久看片| 91麻豆精品国产自产在线游戏 | 西西444www大胆无视频 | 狠狠色丁香久久婷婷综合_中 | 亚洲黄色av网址 | 成人免费共享视频 | 久久精品视 | 国产精品专区在线 | 久草| 91麻豆传媒 | 午夜神马福利 | 91在线永久 | 久草com | 亚洲成人影音 | 8x成人在线 | 99热日本 | 精品福利网站 | 免费av免费观看 | 婷婷色在线播放 | 中文字幕 第二区 | 探花视频免费观看 | 亚洲综合在线五月 | 500部大龄熟乱视频 欧美日本三级 | 激情视频免费在线 | 69亚洲视频 | 激情导航 | 久久试看 | 午夜电影久久久 | 97精品电影院 | 久草在线电影网 | 国产精品免费一区二区 | 欧美一二三区在线观看 | 成人免费大片黄在线播放 | 国产精品18久久久久久首页狼 | 人人干狠狠干 | 久久精品视频在线播放 | 狠狠色噜噜狠狠狠狠2022 | 久久久久久国产精品 | av福利网址导航大全 | 8090yy亚洲精品久久 | 亚洲精品麻豆 | 久久久一本精品99久久精品 | 日韩二区三区在线 | 在线免费亚洲 | 国产欧美日韩精品一区二区免费 | 久久国产三级 | 精品美女在线视频 | 97电影院网 | 国产精品久久久久久久久久ktv | 免费欧美高清视频 | 国产精品原创在线 | 91精品国产99久久久久久红楼 | 国产在线观看高清视频 | 91中文字幕网 | 精品久久久久亚洲 | 91av在线免费看 | 欧美日韩高清 | 国产短视频在线播放 | 日韩网页| 四虎影视欧美 | 久久不射电影网 | 国产精品免费一区二区 | 亚洲涩综合 | 精品久久久精品 | 欧美福利久久 | 98精品国产自产在线观看 | 欧美成a人片在线观看久 | 9在线观看免费高清完整版在线观看明 | 国产精品人人做人人爽人人添 | 天天综合天天做天天综合 | 国产 中文 日韩 欧美 | 香蕉视频免费看 | 免费网站黄 | 日日干夜夜草 | 成人国产综合 | 中文字幕视频播放 | 国产淫片 | 亚洲美女免费精品视频在线观看 | 三级av在线| 亚洲 欧洲 国产 日本 综合 | 久久无码av一区二区三区电影网 | 在线99视频 | 精品日韩视频 | 成人黄色电影在线观看 | 中文字幕亚洲高清 | 国产在线2020 | 免费情缘 | 婷婷视频在线播放 | 亚洲精品2区 | 色婷五月| 激情五月播播久久久精品 | 婷婷久久五月 | 99国产精品久久久久久久久久 | 成人国产精品久久久久久亚洲 | 97理论片 | 操操碰 | 色噜噜日韩精品欧美一区二区 | 四月婷婷在线观看 | 亚洲精品免费播放 | 看片的网址 | 天天操天天艹 | 最近字幕在线观看第一季 | 久久艹在线 | 欧美激情精品久久久久久变态 | 激情婷婷在线 | 久久精品国产亚洲aⅴ | 中文字幕亚洲精品在线观看 | 欧美精品免费在线观看 | 成人av电影网址 | 国产欧美精品在线观看 | 欧美激情视频在线观看免费 | 免费试看一区 | 国内精品亚洲 | 国产精品永久久久久久久www | 国产第一页在线播放 | 婷婷丁香综合 | av在线a| 啪啪小视频网站 | 在线观看你懂的网站 | 亚洲伦理中文字幕 | 中文字幕专区高清在线观看 | 日韩精品不卡在线观看 | 在线免费性生活片 | 国产一区高清在线观看 | 香蕉一区 | 日韩高清免费无专码区 | 国产剧情在线一区 | 国产精品地址 | 99久久网站| 久久综合狠狠综合久久综合88 | 久久精品香蕉 | 一区二区精品在线 | 黄色三级免费网址 | 中文字幕日韩伦理 | 99这里精品 | 日韩av片免费在线观看 | 在线综合 亚洲 欧美在线视频 | 91精品老司机久久一区啪 | 免费av的网站| 美女久久久久 | 2023国产精品自产拍在线观看 | 亚洲国产三级在线 | 亚洲国产精品电影在线观看 | 91视频 - 88av | 午夜精品久久久久久久久久久久 | 97国产一区二区 | 激情欧美xxxx| 色爱区综合激月婷婷 | 免费www视频| 午夜精品电影 | 伊人天堂网 | 日本精品一区二区三区在线播放视频 | 亚洲男模gay裸体gay | 国产在线色视频 | 天天爽夜夜爽人人爽一区二区 | 91亚洲国产成人久久精品网站 | 天天操,夜夜操 | 日韩视频免费在线观看 | 久久久久久国产精品999 | 国产91在线播放 | 国产原厂视频在线观看 | 最新国产精品久久精品 | 在线中文字幕网站 | 一级成人免费视频 | 91视频久久久久久 | 日韩av黄 | 人人草在线视频 | 国产麻豆果冻传媒在线观看 | 久久国产精品色婷婷 | 九九99靖品 | 亚洲午夜久久久久久久久 | 国产国产人免费人成免费视频 | a天堂中文在线 | 成人国产精品免费 | 久久综合九色综合欧美狠狠 | 亚洲天堂网站 | 欧美精品一区二区在线播放 | 人人射网站| 国产在线91精品 | 国产亚洲精品免费 | 国产视频一区二区在线观看 | 国产精品入口66mio女同 | 免费97视频 | 久久综合网色—综合色88 | 超碰在线国产 | 亚洲久草网 | 香蕉视频网址 | 日韩中文字幕在线观看 | 久久不射影院 | 国产一区福利在线 | 久久久福利 | 婷婷综合伊人 | 福利视频入口 | 丁香六月天婷婷 | 久久在线一区 | 日韩欧美精品一区二区三区经典 | 国产精品网站 | 国产系列在线观看 | 91久久久久久久一区二区 | 91精品国自产在线偷拍蜜桃 | 久久久久久综合网天天 | 成人h动漫在线看 | 一级成人免费 | 在线之家免费在线观看电影 | a级片久久久 | 蜜臀av夜夜澡人人爽人人桃色 | 麻豆视频网址 | 欧美福利久久 | 欧美婷婷综合 | 在线观看岛国 | 精品99久久| 天天射夜夜爽 | 99人成在线观看视频 | 丁香婷婷基地 | 国产一区二区三区高清播放 | 婷婷性综合 | 亚洲国产精品久久 | 成人xxxx| 欧美日韩国产一区二区三区在线观看 | 国产精品免费麻豆入口 | 免费看黄色小说的网站 | 91理论片午午伦夜理片久久 | 欧美一二三视频 | 国产精品h在线观看 | 久久视频国产精品免费视频在线 | 91亚洲精品国偷拍自产在线观看 | 久久久久亚洲精品成人网小说 | 国产又粗又硬又爽视频 | 色精品视频 | 久久免费久久 | 免费99视频 | 天天躁日日躁狠狠躁av麻豆 | 国产裸体永久免费视频网站 | 国产成人精品一区二区在线 | 国产一区二区三区在线 | 欧美aⅴ在线观看 | 亚洲综合欧美日韩狠狠色 | 日韩精品一区二区久久 | 亚洲国产精品99久久久久久久久 | 欧美日韩视频一区二区 | 久久小视频 | 久久福利影视 | 国产精品第一页在线观看 | 九九久久婷婷 | 永久免费av在线播放 | 国产精品美女免费看 | 香蕉在线播放 | 人人澡视频| 91人人干 | 亚洲综合激情小说 | 91在线视频精品 | 久久国产精品免费一区二区三区 | 久视频在线播放 | 国产爽视频 | 91在线公开视频 | 国产亚洲精品久久久久久久久久久久 | 久久99精品久久久久婷婷 | www.在线观看视频 | 99久久久国产精品 | 中文字幕在线看视频 | 欧美成人日韩 | 欧美一级视频一区 | 中文字幕在线看视频国产中文版 | 九九热re| 午夜精品电影 | 99视频精品 | 久久久久久免费网 | 九九热只有这里有精品 | 精品一区二区电影 | 久久伊人免费视频 | 国产成人三级在线播放 | 成人中心免费视频 | 亚洲精品黄色片 | 久久久久国产一区二区三区四区 | 麻豆国产精品永久免费视频 | 在线不卡视频 | 日韩免费一区二区 | 色噜噜在线观看 | 五月天综合婷婷 | 日韩欧美一区二区在线观看 | 色综合久久五月天 | 国产正在播放 | 综合视频在线 | 少妇精69xxtheporn | 一区二区三区高清 | 人人爽人人片 | 久久久国产99久久国产一 | 日韩二区在线观看 | 亚州av网站 | 中文字幕免费久久 | 91精品国产麻豆国产自产影视 | 五月婷婷婷婷婷 | 国产高清精 | 国产99区| 国产偷国产偷亚洲清高 | 日日骑 | 超碰97网站 | 91 在线视频播放 | 国产成人av网 | 九九日韩| 久久伊人综合 | 91精品国产综合久久福利不卡 | 日本久久免费视频 | 亚洲精品欧美精品 | 精品成人在线 | av成人在线网站 | 久草精品在线播放 | 亚洲一区二区视频 | 最近中文字幕久久 | 最近中文字幕免费大全 | 91麻豆精品91久久久久同性 | 日韩在线免费高清视频 | 黄色亚洲在线 | 五月天激情综合 | 国产视频一区在线播放 | 美女视频久久 | 欧美国产在线看 | 伊人网av | 伊人永久| 国产亚洲精品av | 日本成人中文字幕在线观看 | 狠狠操夜夜操 | 婷婷综合导航 | 色99在线| 精品久久久免费视频 | 五月婷影院| 国产理论免费 | 中文字幕高清在线 | 人成午夜视频 | 91精品综合在线观看 | 激情综合网色播五月 | 去看片 | 99久久精品久久久久久清纯 | 青青看片 | 夜夜夜| 99性视频 | 少妇激情久久 | 天天色天天操天天爽 | 亚洲欧洲精品一区二区精品久久久 | 亚洲精品一区二区三区四区高清 | 成人一区二区三区中文字幕 | 亚洲视频在线观看 | 成人在线一区二区 | 久久久久女人精品毛片 | 日韩中字在线观看 | 2022国产精品视频 | 综合网成人 | 久久草 | 黄色免费在线视频 | 日韩视频一二三区 | 国产自产高清不卡 | 国产91aaa | 欧美日韩天堂 | 日韩精品视频久久 | 波多野结衣小视频 | 国产精品久久久久一区二区三区共 | 综合精品久久久 | 美女网站在线播放 | 国产精品久久久久久久久久久杏吧 | 国产小视频你懂的在线 | 精品网站999www | 在线色网站 | 国产精品毛片一区视频播不卡 | 奇米影视四色8888 | 国产精品男女 | 深爱开心激情网 | 国色天香在线观看 | 国产精久久久久久妇女av | 日韩a在线看 | 日韩午夜电影院 | 一区二区三区在线电影 | 国产亚洲精品久久网站 | 97在线精品国自产拍中文 | 亚洲精品美女久久久久 | 久久国产高清 | 免费看的黄网站软件 | 国产 日韩 中文字幕 | 麻豆一区二区 | 91亚洲精品久久久蜜桃网站 | 综合色伊人 | 天天射天天射天天 | 成人毛片在线观看视频 | 成人性生交大片免费观看网站 | 97**国产露脸精品国产 | 国产精品永久久久久久久久久 | 国产黄a三级 | 97超碰伊人| 国产一区二区三区免费视频 | 亚洲精品久久久久中文字幕二区 | 久久成人亚洲欧美电影 | 国产老太婆免费交性大片 | 国产亚洲人成网站在线观看 | 精品久久久久久久久久久久久 | 欧美日韩国产一区二区三区在线观看 | 天天综合日日夜夜 | 国产黄色理论片 | 国产成人在线精品 | 国产乱对白刺激视频在线观看女王 | www.天天操| 久久精品日产第一区二区三区乱码 | 日日干网 | 亚洲精品视 | 日韩av资源站 | 欧美一级欧美一级 | 99精品免费网 | 久草手机视频 | 五月婷婷激情六月 | 91成年视频 | 国产91精品久久久久 | 久久资源在线 | www.国产视频 | 偷拍精品一区二区三区 | www色综合 | 黄色免费网战 | 国产精品毛片一区二区三区 | 91成品视频 | 在线亚洲欧美日韩 | 亚洲草视频 | 欧美性久久久 | 怡春院av| 18岁免费看片 | 国内精品久久久久影院一蜜桃 | 亚洲在线视频免费 | av免费黄色 | 色综合久久久久 | 天天艹天天操 | 在线观看你懂的网址 | 日本最新一区二区三区 | 精品国产乱码久久久久 | 免费看短 | 欧美少妇xxxxxx | 亚洲精品视频观看 | 黄色免费电影网站 | 欧美日韩三级 | 91亚洲精品在线观看 | 69av视频在线 | 日韩三级不卡 | 深爱激情亚洲 | 国产麻豆果冻传媒在线观看 | 国产成人精品一区二区三区福利 | www.91国产 | 精品一区二区在线免费观看 | 2024国产在线 | 亚洲传媒在线 | 天堂在线v| 丝袜网站在线观看 | 国产999视频 | 91爱爱视频| 日韩视频欧美视频 | www.天堂av| 三级小视频在线观看 | 91九色视频网站 | 免费在线观看成人 | 久久久久免费精品视频 | 97在线观看免费高清完整版在线观看 | 欧美国产日韩一区二区三区 | 正在播放 国产精品 | 亚洲午夜精品久久久久久久久久久久 | 日韩午夜在线观看 | av一级网站 | av福利在线看 | 国产日韩欧美在线一区 | 一本一本久久aa综合精品 | 久久久久亚洲精品男人的天堂 | 韩国av免费观看 | 三三级黄色片之日韩 | 麻豆一级视频 | 91视频久久久久久 | 国产精品视频最多的网站 | 国产精品久久久久婷婷 | 国产 成人 久久 | 日日夜夜骑 | 97在线精品| 成人免费观看a | 91精品久久久久久久91蜜桃 | 在线观看视频在线观看 | 美女啪啪图片 | av高清免费 | 国产精品久久久久久久久久东京 | 在线观看蜜桃视频 | 久久久精品国产一区二区三区 | 有码中文字幕在线观看 | 99久久久国产精品免费观看 | 天天爱天天操天天爽 | 久久久精品久久日韩一区综合 | 91成熟丰满女人少妇 | 久久新视频 | 婷婷伊人综合 | 91国内在线 | 成年人视频在线免费播放 | 日韩成人精品一区二区三区 | 国产一区二区在线免费观看 | 人人看97| 免费网站观看www在线观看 | 一区二区三区中文字幕在线 | www.夜夜操 | 亚洲在线视频免费 | 91精品在线视频观看 | 狠狠色丁香婷婷综合最新地址 | 午夜av剧场 | 日韩一级成人av | 超碰com| 深爱婷婷久久综合 | 毛片永久新网址首页 | 69绿帽绿奴3pvideos| 色多多视频在线观看 | 欧美性色黄大片在线观看 | 亚洲精品1234区 | 综合久久网站 | 美女久久99| 超碰97免费| 久久精品爱爱视频 | 久久久久在线观看 | 久久久久看片 | 综合网中文字幕 | 99精品国产亚洲 | 亚洲视频久久久久 | 久久综合福利 | 久久这里只有精品视频99 | 中文字幕亚洲不卡 | 亚洲国产理论片 | 狠狠干天天射 | 久久久精品视频网站 | www.在线观看视频 | 九色视频网站 | 亚洲精品国产免费 | 日本三级国产 | 欧美午夜久久久 | 2019中文在线观看 | 天天插综合 | 日韩视频一 | 日韩免费一区二区在线观看 | 最新色站 | 日韩高清在线一区二区三区 | 夜夜躁狠狠躁日日躁视频黑人 | 日韩免费中文 | 久久久久亚洲精品国产 | 欧美日韩精品在线免费观看 | 亚州成人av在线 | 中文字幕美女免费在线 | 五月天最新网址 | 成人丁香花| 日韩色综合网 | 久久久久婷 | 91精品视频免费 | 午夜aaaa| 国内一区二区视频 | 久久久久久国产精品亚洲78 | 久久久久精 | 久久九精品 | 色网站视频 | 97精产国品一二三产区在线 | 综合色站导航 | 国产视频一区二区在线播放 | 五月婷婷色 | 97超碰中文 | 久久人人爽人人爽人人片av免费 | 狠狠插狠狠操 | 亚洲人成人天堂h久久 | 久久久久黄 | 国产一区在线精品 | 亚洲综合一区二区精品导航 | 国产激情小视频在线观看 | 97成人精品视频在线观看 | 国产喷水在线 | 免费成人在线电影 | 精品国产区 | 国产破处视频在线播放 | 欧美日韩亚洲精品在线 | 久久在线影院 | 一区二区三区四区在线 | 日韩欧美电影在线观看 | 91免费国产在线观看 | 2020天天干夜夜爽 | 91看片淫黄大片在线播放 | 日韩在线免费视频观看 | 久久久久久久久久电影 | 香蕉视频4aa| 国产成人av福利 | www视频在线免费观看 | 国产精品综合久久久久久 | 五月天综合激情 | 人人爽人人爽人人片av | 欧美日一级片 | 天天干天天操天天拍 | 国产午夜三级一区二区三桃花影视 | 91九色蝌蚪在线 | 97成人精品 | 天天射天天爽 | 免费看污片 | 波多在线视频 | 中文一区在线观看 | 国产精品1区2区 | 日韩一区在线播放 | 欧美成人手机版 | 五月花丁香婷婷 | 在线色亚洲 | 黄影院| 日本美女xx | 免费观看午夜视频 | 欧洲在线免费视频 | 在线观看www.| 午夜av在线播放 | 成年人在线观看免费视频 | 天天射天天操天天 | 色综合色综合久久综合频道88 | 一区二区三区免费在线观看视频 | 色99色| 91.麻豆视频| 国产精品黑丝在线观看 | 欧美性生活久久 | 在线国产专区 | 韩国精品在线观看 | 欧美一级黄色视屏 | 日本韩国中文字幕 | 日韩欧美精品免费 | 一级做a视频 | 狠狠色丁香婷婷综合视频 | 亚洲黄色片一级 | 欧美国产精品久久久久久免费 | 成人黄色在线看 | 国产高清在线不卡 | 欧美日韩免费观看一区二区三区 | 中文字幕在线免费观看视频 | 久久精品爱视频 | 欧美一二三区在线观看 | 91在线视频网址 | 最新中文字幕在线观看视频 | 黄色网www | www.久久视频 | 久久精品视频播放 | 国产96在线视频 | 国产破处在线播放 | 国产精品一区二区三区四区在线观看 | 西西大胆啪啪 | 91在线超碰| 99日韩精品 | 国产高清在线观看 | 一级黄网| 国产99久久九九精品免费 | 91成人精品一区在线播放69 | 噜噜色官网 | 人人爽爽人人 | 青青草国产精品视频 | 欧美片一区二区三区 | 91看片在线观看 | 久久综合综合久久综合 | 成人av影视观看 | 特级黄色一级 | www,黄视频 | 色干干| 在线精品在线 | 91av网址 | 精品国产视频一区 | 国产一级二级在线播放 | 人人干网| 午夜精品视频福利 | 国产精品va在线观看入 | 国内精品久久久精品电影院 | 在线黄色av | 一区二区三区在线影院 | 欧美黄污视频 | 国产日产高清dvd碟片 | 99久久国产免费,99久久国产免费大片 | 亚洲精品2区 | 在线免费视 | 黄网站免费大全入口 | 久久99精品久久久久久秒播蜜臀 | 成人免费视频免费观看 | www.xxxx欧美 | 亚洲网站在线 | 天天爽天天碰狠狠添 | 六月丁香在线观看 | 日韩在线观看a | 九九视频在线观看视频6 | 久久久精品国产一区二区电影四季 | 国产啊v在线观看 | 亚洲人成网站精品片在线观看 | 日韩在线观看网站 | 九九热av | 丝袜护士aⅴ在线白丝护士 天天综合精品 | 色噜噜日韩精品一区二区三区视频 | 日韩三级中文字幕 | 欧美午夜视频在线 | 中文字幕影片免费在线观看 | 久久成人国产精品入口 | 欧美日韩中文字幕综合视频 | 亚洲专区中文字幕 | 亚洲免费在线 | 久视频在线播放 | 日本性视频 | 在线免费黄 | 欧美视屏一区二区 | 日韩三级视频在线观看 | 一级国产视频 | 午夜91视频 | 中文字幕在线日本 | 久久婷婷五月综合色丁香 | 99在线热播精品免费 | 亚洲成aⅴ人片久久青草影院 | 亚洲欧美在线视频免费 | 国产露脸91国语对白 | 久久深夜福利免费观看 | 亚洲视屏一区 | 韩国精品一区二区三区六区色诱 | 色综合天天综合在线视频 | 在线观看一级 | 国产色爽 | 不卡的av在线播放 | 不卡中文字幕av | 色噜噜色噜噜 | 在线国产视频一区 | av大片免费在线观看 | 久久福利小视频 | 乱子伦av | av黄免费看 | aaa黄色毛片 | 911精品美国片911久久久 | 久久精品视频国产 | 亚洲精品视频免费观看 | 国产精品乱码久久 | 天天曰 | 久久福利在线 | 久久99精品久久只有精品 | 黄色免费网站下载 | 免费av网站在线看 | 精品产品国产在线不卡 | 免费在线观看一级片 | 91亚洲精品国偷拍自产在线观看 | 色综合久久天天 | 激情丁香| 亚洲精品tv久久久久久久久久 | 精品亚洲va在线va天堂资源站 | 国产婷婷精品 | 亚洲视频免费 | 在线激情网 | 香蕉视频在线观看免费 | 国产 欧美 日韩 | 国产999精品久久久久久绿帽 | 91看片淫黄大片一级在线观看 | 91成人在线视频 | h动漫中文字幕 | 在线免费黄色av | 国产成年人av | 国产一级免费播放 | 一级黄色av| 国产精品中文 | 69精品人人人人 | 成人av电影在线 | 99免费观看视频 | 不卡的av在线播放 | 日韩欧美91 | 国产一区播放 | 视频在线精品 | a一片一级 | 日韩免费视频观看 | 久久精品一区二 | 日日弄天天弄美女bbbb | 97在线资源 | 夜夜躁日日躁狠狠久久88av | 亚洲视频一区二区三区在线观看 | 成年人免费在线观看网站 | 日韩精品一区在线观看 | 色视频在线观看 | 一区二区不卡高清 | 91成人午夜 | 久久躁日日躁aaaaxxxx | 婷婷丁香在线 | 私人av| 国产一区二区在线观看视频 | 丁香亚洲 | www.色com | 久久色在线播放 | 国产精品乱码久久久久久1区2区 |