mirror of
https://e.coding.net/weidongshan/linux/doc_and_source_for_drivers.git
synced 2025-12-02 04:41:19 +08:00
add 09_UART/15,16
This commit is contained in:
@@ -0,0 +1,288 @@
|
||||
#include <linux/module.h>
|
||||
#include <linux/ioport.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/console.h>
|
||||
#include <linux/sysrq.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/tty.h>
|
||||
#include <linux/tty_flip.h>
|
||||
#include <linux/serial_core.h>
|
||||
#include <linux/serial.h>
|
||||
#include <linux/clk.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/rational.h>
|
||||
#include <linux/reset.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/io.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
|
||||
#include <asm/irq.h>
|
||||
|
||||
#define BUF_LEN 1024
|
||||
#define NEXT_PLACE(i) ((i+1)&0x3FF)
|
||||
|
||||
static struct uart_port *virt_port;
|
||||
static unsigned char txbuf[BUF_LEN];
|
||||
static int tx_buf_r = 0;
|
||||
static int tx_buf_w = 0;
|
||||
|
||||
static unsigned char rxbuf[BUF_LEN];
|
||||
static int rx_buf_r = 0;
|
||||
static int rx_buf_w = 0;
|
||||
|
||||
static struct proc_dir_entry *uart_proc_file;
|
||||
|
||||
|
||||
static struct uart_driver virt_uart_drv = {
|
||||
.owner = THIS_MODULE,
|
||||
.driver_name = "VIRT_UART",
|
||||
.dev_name = "ttyVIRT",
|
||||
.major = 0,
|
||||
.minor = 0,
|
||||
.nr = 1,
|
||||
};
|
||||
|
||||
|
||||
/* circle buffer */
|
||||
|
||||
|
||||
static int is_txbuf_empty(void)
|
||||
{
|
||||
return tx_buf_r == tx_buf_w;
|
||||
}
|
||||
|
||||
static int is_txbuf_full(void)
|
||||
{
|
||||
return NEXT_PLACE(tx_buf_w) == tx_buf_r;
|
||||
}
|
||||
|
||||
static int txbuf_put(unsigned char val)
|
||||
{
|
||||
if (is_txbuf_full())
|
||||
return -1;
|
||||
txbuf[tx_buf_w] = val;
|
||||
tx_buf_w = NEXT_PLACE(tx_buf_w);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int txbuf_get(unsigned char *pval)
|
||||
{
|
||||
if (is_txbuf_empty())
|
||||
return -1;
|
||||
*pval = g_events[tx_buf_r];
|
||||
tx_buf_r = NEXT_PLACE(tx_buf_r);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int txbuf_count(void)
|
||||
{
|
||||
if (tx_buf_w >= tx_buf_r)
|
||||
return tx_buf_w - tx_buf_r;
|
||||
else
|
||||
return BUF_LEN + tx_buf_w - tx_buf_r;
|
||||
}
|
||||
|
||||
|
||||
|
||||
ssize_t virt_uart_buf_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
|
||||
{
|
||||
/* 把txbuf中的数据copy_to_user */
|
||||
int cnt = txbuf_count();
|
||||
int i;
|
||||
unsigned char val;
|
||||
|
||||
cnt = (cnt > size)? size: cnt;
|
||||
|
||||
for (i = 0; i < cnt; i++)
|
||||
{
|
||||
txbuf_get(&val);
|
||||
copy_to_user(buf+i, &val, 1);
|
||||
}
|
||||
|
||||
return cnt;
|
||||
}
|
||||
|
||||
static ssize_t virt_uart_buf_write (struct file *file, const char __user *buf, size_t size, loff_t *off)
|
||||
{
|
||||
/* get data */
|
||||
copy_from_user(rxbuf, buf, size);
|
||||
rx_buf_w = size;
|
||||
|
||||
/* 模拟产生RX中断 */
|
||||
irq_set_irqchip_state(virt_port->irq, IRQCHIP_STATE_PENDING, 1);
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
static const struct file_operations virt_uart_buf_fops = {
|
||||
.read = virt_uart_buf_read,
|
||||
.write = virt_uart_buf_write,
|
||||
};
|
||||
|
||||
|
||||
static unsigned int virt_tx_empty(struct uart_port *port)
|
||||
{
|
||||
/* 因为要发送的数据瞬间存入buffer */
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* interrupts disabled on entry
|
||||
*/
|
||||
static void virt_start_tx(struct uart_port *port)
|
||||
{
|
||||
struct circ_buf *xmit = &port->state->xmit;
|
||||
|
||||
while (!uart_circ_empty(xmit) &&
|
||||
!uart_tx_stopped(port)) {
|
||||
/* send xmit->buf[xmit->tail]
|
||||
* out the port here */
|
||||
|
||||
/* 把circ buffer中的数据全部存入txbuf */
|
||||
|
||||
//txbuf[tx_buf_w++] = xmit->buf[xmit->tail];
|
||||
txbuf_put(xmit->buf[xmit->tail]);
|
||||
xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
|
||||
port->icount.tx++;
|
||||
}
|
||||
|
||||
if (uart_circ_chars_pending(xmit) < WAKEUP_CHARS)
|
||||
uart_write_wakeup(port);
|
||||
|
||||
}
|
||||
|
||||
static void
|
||||
virt_set_termios(struct uart_port *port, struct ktermios *termios,
|
||||
struct ktermios *old)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
static const struct uart_ops virt_pops = {
|
||||
.tx_empty = virt_tx_empty,
|
||||
//.set_mctrl = imx_set_mctrl,
|
||||
//.get_mctrl = imx_get_mctrl,
|
||||
//.stop_tx = imx_stop_tx,
|
||||
.start_tx = virt_start_tx,
|
||||
//.stop_rx = imx_stop_rx,
|
||||
//.enable_ms = imx_enable_ms,
|
||||
//.break_ctl = imx_break_ctl,
|
||||
//.startup = imx_startup,
|
||||
//.shutdown = imx_shutdown,
|
||||
//.flush_buffer = imx_flush_buffer,
|
||||
.set_termios = virt_set_termios,
|
||||
//.type = imx_type,
|
||||
//.config_port = imx_config_port,
|
||||
//.verify_port = imx_verify_port,
|
||||
};
|
||||
|
||||
|
||||
static irqreturn_t virt_uart_rxint(int irq, void *dev_id)
|
||||
{
|
||||
struct uart_port *port = dev_id;
|
||||
unsigned long flags;
|
||||
int i;
|
||||
|
||||
spin_lock_irqsave(&port->lock, flags);
|
||||
|
||||
for (i = 0; i < rx_buf_w; i++) {
|
||||
port->icount.rx++;
|
||||
|
||||
/* get data from hardware/rxbuf */
|
||||
|
||||
/* put data to ldisc */
|
||||
tty_insert_flip_char(port, rxbuf[i], TTY_NORMAL)
|
||||
}
|
||||
rx_buf_w = 0;
|
||||
|
||||
spin_unlock_irqrestore(&port->lock, flags);
|
||||
tty_flip_buffer_push(port);
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
|
||||
static int virtual_uart_probe(struct platform_device *pdev)
|
||||
{
|
||||
int rxirq;
|
||||
int ret;
|
||||
|
||||
/* create proc file */
|
||||
uart_proc_file = proc_create("virt_uart_buf", 0, NULL, &virt_uart_buf_fops);
|
||||
|
||||
//uart_add_one_port(struct uart_driver * drv, struct uart_port * uport);
|
||||
|
||||
/* 从设备树获得硬件信息 */
|
||||
rxirq = platform_get_irq(pdev, 0);
|
||||
ret = devm_request_irq(&pdev->dev, rxirq, virt_uart_rxint, 0,
|
||||
dev_name(&pdev->dev), virt_port);
|
||||
|
||||
/* 分配设置注册uart_port */
|
||||
virt_port = devm_kzalloc(&pdev->dev, sizeof(*virt_port), GFP_KERNEL);
|
||||
|
||||
virt_port->dev = &pdev->dev;
|
||||
virt_port->iotype = UPIO_MEM;
|
||||
virt_port->irq = rxirq;
|
||||
virt_port->fifosize = 32;
|
||||
virt_port->ops = &virt_pops;
|
||||
virt_port->flags = UPF_BOOT_AUTOCONF;
|
||||
|
||||
return uart_add_one_port(&virt_uart_drv, virt_port);
|
||||
|
||||
}
|
||||
static int virtual_uart_remove(struct platform_device *pdev)
|
||||
{
|
||||
proc_remove(uart_proc_file);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
static const struct of_device_id virtual_uart_of_match[] = {
|
||||
{ .compatible = "100ask,virtual_uart", },
|
||||
{ },
|
||||
};
|
||||
|
||||
|
||||
static struct platform_driver virtual_uart_driver = {
|
||||
.probe = virtual_uart_probe,
|
||||
.remove = virtual_uart_remove,
|
||||
.driver = {
|
||||
.name = "100ask_virtual_uart",
|
||||
.of_match_table = of_match_ptr(virtual_uart_of_match),
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
/* 1. 入口函数 */
|
||||
static int __init virtual_uart_init(void)
|
||||
{
|
||||
printk("%s %s %d\n", __FILE__, __FUNCTION__, __LINE__);
|
||||
|
||||
int ret = uart_register_driver(&virt_uart_drv);
|
||||
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* 1.1 注册一个platform_driver */
|
||||
return platform_driver_register(&virtual_uart_driver);
|
||||
}
|
||||
|
||||
|
||||
/* 2. 出口函数 */
|
||||
static void __exit virtual_uart_exit(void)
|
||||
{
|
||||
printk("%s %s %d\n", __FILE__, __FUNCTION__, __LINE__);
|
||||
/* 2.1 反注册platform_driver */
|
||||
platform_driver_unregister(&virtual_uart_driver);
|
||||
}
|
||||
|
||||
module_init(virtual_uart_init);
|
||||
module_exit(virtual_uart_exit);
|
||||
|
||||
MODULE_LICENSE("GPL");
|
||||
|
||||
|
||||
@@ -0,0 +1,13 @@
|
||||
|
||||
|
||||
/{
|
||||
virtual_uart: virtual_uart_100ask {
|
||||
compatible = "100ask,virtual_uart";
|
||||
|
||||
interrupt-parent = <&intc>;
|
||||
interrupts = <GIC_SPI 99 IRQ_TYPE_LEVEL_HIGH>;
|
||||
|
||||
};
|
||||
|
||||
|
||||
};
|
||||
20
IMX6ULL/source/09_UART/07_virtual_uart_driver_ok/Makefile
Normal file
20
IMX6ULL/source/09_UART/07_virtual_uart_driver_ok/Makefile
Normal file
@@ -0,0 +1,20 @@
|
||||
|
||||
# 1. 使用不同的开发板内核时, 一定要修改KERN_DIR
|
||||
# 2. KERN_DIR中的内核要事先配置、编译, 为了能编译内核, 要先设置下列环境变量:
|
||||
# 2.1 ARCH, 比如: export ARCH=arm64
|
||||
# 2.2 CROSS_COMPILE, 比如: export CROSS_COMPILE=aarch64-linux-gnu-
|
||||
# 2.3 PATH, 比如: export PATH=$PATH:/home/book/100ask_roc-rk3399-pc/ToolChain-6.3.1/gcc-linaro-6.3.1-2017.05-x86_64_aarch64-linux-gnu/bin
|
||||
# 注意: 不同的开发板不同的编译器上述3个环境变量不一定相同,
|
||||
# 请参考各开发板的高级用户使用手册
|
||||
|
||||
KERN_DIR = /home/book/100ask_imx6ull-sdk/Linux-4.9.88
|
||||
|
||||
all:
|
||||
make -C $(KERN_DIR) M=`pwd` modules
|
||||
|
||||
clean:
|
||||
make -C $(KERN_DIR) M=`pwd` modules clean
|
||||
rm -rf modules.order
|
||||
|
||||
obj-m += virtual_uart.o
|
||||
|
||||
334
IMX6ULL/source/09_UART/07_virtual_uart_driver_ok/virtual_uart.c
Normal file
334
IMX6ULL/source/09_UART/07_virtual_uart_driver_ok/virtual_uart.c
Normal file
@@ -0,0 +1,334 @@
|
||||
#include <linux/module.h>
|
||||
#include <linux/ioport.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/console.h>
|
||||
#include <linux/sysrq.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/tty.h>
|
||||
#include <linux/tty_flip.h>
|
||||
#include <linux/serial_core.h>
|
||||
#include <linux/serial.h>
|
||||
#include <linux/clk.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/rational.h>
|
||||
#include <linux/reset.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_device.h>
|
||||
#include <linux/io.h>
|
||||
#include <linux/dma-mapping.h>
|
||||
#include <linux/proc_fs.h>
|
||||
|
||||
#include <asm/irq.h>
|
||||
|
||||
#define BUF_LEN 1024
|
||||
#define NEXT_PLACE(i) ((i+1)&0x3FF)
|
||||
|
||||
static struct uart_port *virt_port;
|
||||
static unsigned char txbuf[BUF_LEN];
|
||||
static int tx_buf_r = 0;
|
||||
static int tx_buf_w = 0;
|
||||
|
||||
static unsigned char rxbuf[BUF_LEN];
|
||||
static int rx_buf_w = 0;
|
||||
|
||||
static struct proc_dir_entry *uart_proc_file;
|
||||
|
||||
|
||||
static struct uart_driver virt_uart_drv = {
|
||||
.owner = THIS_MODULE,
|
||||
.driver_name = "VIRT_UART",
|
||||
.dev_name = "ttyVIRT",
|
||||
.major = 0,
|
||||
.minor = 0,
|
||||
.nr = 1,
|
||||
};
|
||||
|
||||
|
||||
/* circle buffer */
|
||||
|
||||
|
||||
static int is_txbuf_empty(void)
|
||||
{
|
||||
return tx_buf_r == tx_buf_w;
|
||||
}
|
||||
|
||||
static int is_txbuf_full(void)
|
||||
{
|
||||
return NEXT_PLACE(tx_buf_w) == tx_buf_r;
|
||||
}
|
||||
|
||||
static int txbuf_put(unsigned char val)
|
||||
{
|
||||
if (is_txbuf_full())
|
||||
return -1;
|
||||
txbuf[tx_buf_w] = val;
|
||||
tx_buf_w = NEXT_PLACE(tx_buf_w);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int txbuf_get(unsigned char *pval)
|
||||
{
|
||||
if (is_txbuf_empty())
|
||||
return -1;
|
||||
*pval = txbuf[tx_buf_r];
|
||||
tx_buf_r = NEXT_PLACE(tx_buf_r);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int txbuf_count(void)
|
||||
{
|
||||
if (tx_buf_w >= tx_buf_r)
|
||||
return tx_buf_w - tx_buf_r;
|
||||
else
|
||||
return BUF_LEN + tx_buf_w - tx_buf_r;
|
||||
}
|
||||
|
||||
|
||||
|
||||
ssize_t virt_uart_buf_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
|
||||
{
|
||||
/* 把txbuf中的数据copy_to_user */
|
||||
int cnt = txbuf_count();
|
||||
int i;
|
||||
unsigned char val;
|
||||
int ret;
|
||||
|
||||
cnt = (cnt > size)? size: cnt;
|
||||
|
||||
for (i = 0; i < cnt; i++)
|
||||
{
|
||||
txbuf_get(&val);
|
||||
ret = copy_to_user(buf+i, &val, 1);
|
||||
}
|
||||
|
||||
return cnt;
|
||||
}
|
||||
|
||||
static ssize_t virt_uart_buf_write (struct file *file, const char __user *buf, size_t size, loff_t *off)
|
||||
{
|
||||
int ret;
|
||||
|
||||
/* get data */
|
||||
ret = copy_from_user(rxbuf, buf, size);
|
||||
rx_buf_w = size;
|
||||
|
||||
/* 模拟产生RX中断 */
|
||||
irq_set_irqchip_state(virt_port->irq, IRQCHIP_STATE_PENDING, 1);
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
static const struct file_operations virt_uart_buf_fops = {
|
||||
.read = virt_uart_buf_read,
|
||||
.write = virt_uart_buf_write,
|
||||
};
|
||||
|
||||
|
||||
static unsigned int virt_tx_empty(struct uart_port *port)
|
||||
{
|
||||
/* 因为要发送的数据瞬间存入buffer */
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* interrupts disabled on entry
|
||||
*/
|
||||
static void virt_start_tx(struct uart_port *port)
|
||||
{
|
||||
struct circ_buf *xmit = &port->state->xmit;
|
||||
|
||||
while (!uart_circ_empty(xmit) &&
|
||||
!uart_tx_stopped(port)) {
|
||||
/* send xmit->buf[xmit->tail]
|
||||
* out the port here */
|
||||
|
||||
/* 把circ buffer中的数据全部存入txbuf */
|
||||
|
||||
//txbuf[tx_buf_w++] = xmit->buf[xmit->tail];
|
||||
txbuf_put(xmit->buf[xmit->tail]);
|
||||
xmit->tail = (xmit->tail + 1) & (UART_XMIT_SIZE - 1);
|
||||
port->icount.tx++;
|
||||
}
|
||||
|
||||
if (uart_circ_chars_pending(xmit) < WAKEUP_CHARS)
|
||||
uart_write_wakeup(port);
|
||||
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
virt_set_termios(struct uart_port *port, struct ktermios *termios,
|
||||
struct ktermios *old)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
static int virt_startup(struct uart_port *port)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void virt_set_mctrl(struct uart_port *port, unsigned int mctrl)
|
||||
{
|
||||
}
|
||||
|
||||
static unsigned int virt_get_mctrl(struct uart_port *port)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void virt_stop_tx(struct uart_port *port)
|
||||
{
|
||||
}
|
||||
|
||||
static void virt_stop_rx(struct uart_port *port)
|
||||
{
|
||||
}
|
||||
|
||||
static void virt_shutdown(struct uart_port *port)
|
||||
{
|
||||
}
|
||||
|
||||
static const char *virt_type(struct uart_port *port)
|
||||
{
|
||||
return "100ASK_VIRT_UART";
|
||||
}
|
||||
|
||||
|
||||
static const struct uart_ops virt_pops = {
|
||||
.tx_empty = virt_tx_empty,
|
||||
.set_mctrl = virt_set_mctrl,
|
||||
.get_mctrl = virt_get_mctrl,
|
||||
.stop_tx = virt_stop_tx,
|
||||
.start_tx = virt_start_tx,
|
||||
.stop_rx = virt_stop_rx,
|
||||
//.enable_ms = imx_enable_ms,
|
||||
//.break_ctl = imx_break_ctl,
|
||||
.startup = virt_startup,
|
||||
.shutdown = virt_shutdown,
|
||||
//.flush_buffer = imx_flush_buffer,
|
||||
.set_termios = virt_set_termios,
|
||||
.type = virt_type,
|
||||
//.config_port = imx_config_port,
|
||||
//.verify_port = imx_verify_port,
|
||||
};
|
||||
|
||||
|
||||
|
||||
static irqreturn_t virt_uart_rxint(int irq, void *dev_id)
|
||||
{
|
||||
struct uart_port *port = dev_id;
|
||||
struct tty_port *tport = &port->state->port;
|
||||
unsigned long flags;
|
||||
int i;
|
||||
|
||||
spin_lock_irqsave(&port->lock, flags);
|
||||
|
||||
for (i = 0; i < rx_buf_w; i++) {
|
||||
port->icount.rx++;
|
||||
|
||||
/* get data from hardware/rxbuf */
|
||||
|
||||
/* put data to ldisc */
|
||||
tty_insert_flip_char(tport, rxbuf[i], TTY_NORMAL);
|
||||
}
|
||||
rx_buf_w = 0;
|
||||
|
||||
spin_unlock_irqrestore(&port->lock, flags);
|
||||
tty_flip_buffer_push(tport);
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
|
||||
static int virtual_uart_probe(struct platform_device *pdev)
|
||||
{
|
||||
int rxirq;
|
||||
int ret;
|
||||
|
||||
/* create proc file */
|
||||
uart_proc_file = proc_create("virt_uart_buf", 0, NULL, &virt_uart_buf_fops);
|
||||
|
||||
//uart_add_one_port(struct uart_driver * drv, struct uart_port * uport);
|
||||
|
||||
/* 从设备树获得硬件信息 */
|
||||
rxirq = platform_get_irq(pdev, 0);
|
||||
|
||||
/* 分配设置注册uart_port */
|
||||
virt_port = devm_kzalloc(&pdev->dev, sizeof(*virt_port), GFP_KERNEL);
|
||||
|
||||
virt_port->dev = &pdev->dev;
|
||||
virt_port->iotype = UPIO_MEM;
|
||||
virt_port->irq = rxirq;
|
||||
virt_port->fifosize = 32;
|
||||
virt_port->ops = &virt_pops;
|
||||
virt_port->flags = UPF_BOOT_AUTOCONF;
|
||||
virt_port->type = PORT_8250;
|
||||
|
||||
ret = devm_request_irq(&pdev->dev, rxirq, virt_uart_rxint, 0,
|
||||
dev_name(&pdev->dev), virt_port);
|
||||
|
||||
return uart_add_one_port(&virt_uart_drv, virt_port);
|
||||
|
||||
}
|
||||
static int virtual_uart_remove(struct platform_device *pdev)
|
||||
{
|
||||
uart_remove_one_port(&virt_uart_drv, virt_port);
|
||||
proc_remove(uart_proc_file);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
static const struct of_device_id virtual_uart_of_match[] = {
|
||||
{ .compatible = "100ask,virtual_uart", },
|
||||
{ },
|
||||
};
|
||||
|
||||
|
||||
static struct platform_driver virtual_uart_driver = {
|
||||
.probe = virtual_uart_probe,
|
||||
.remove = virtual_uart_remove,
|
||||
.driver = {
|
||||
.name = "100ask_virtual_uart",
|
||||
.of_match_table = of_match_ptr(virtual_uart_of_match),
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
/* 1. 入口函数 */
|
||||
static int __init virtual_uart_init(void)
|
||||
{
|
||||
int ret;
|
||||
|
||||
printk("%s %s %d\n", __FILE__, __FUNCTION__, __LINE__);
|
||||
|
||||
ret = uart_register_driver(&virt_uart_drv);
|
||||
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* 1.1 注册一个platform_driver */
|
||||
return platform_driver_register(&virtual_uart_driver);
|
||||
}
|
||||
|
||||
|
||||
/* 2. 出口函数 */
|
||||
static void __exit virtual_uart_exit(void)
|
||||
{
|
||||
printk("%s %s %d\n", __FILE__, __FUNCTION__, __LINE__);
|
||||
/* 2.1 反注册platform_driver */
|
||||
platform_driver_unregister(&virtual_uart_driver);
|
||||
|
||||
|
||||
uart_unregister_driver(&virt_uart_drv);
|
||||
}
|
||||
|
||||
module_init(virtual_uart_init);
|
||||
module_exit(virtual_uart_exit);
|
||||
|
||||
MODULE_LICENSE("GPL");
|
||||
|
||||
|
||||
@@ -0,0 +1,13 @@
|
||||
|
||||
|
||||
/{
|
||||
virtual_uart: virtual_uart_100ask {
|
||||
compatible = "100ask,virtual_uart";
|
||||
|
||||
interrupt-parent = <&intc>;
|
||||
interrupts = <GIC_SPI 99 IRQ_TYPE_LEVEL_HIGH>;
|
||||
|
||||
};
|
||||
|
||||
|
||||
};
|
||||
Reference in New Issue
Block a user