linux uart dma模式,linux-device-driver

我已经在传输模式下为UART编写了一个简单的器件驱动程序,并启用了DMA和中断。 我正在使用的硬件是加载了Linux 3.4的omap 4460 pandaboard。

下面我分享代码的相关部分。 在开放阶段:

dma_map = ioremap(UART4_DMA_REG,0x1350);

if(dma_map == NULL) {

printk(KERN_INFO " unable to io_remap DMA region\n");

return -ENOMEM;

}

printk(KERN_INFO "DMA mapping successful\n");

irq_val = request_irq(45,uart_handler,IRQF_DISABLED,"uart_int",NULL);

if(irq_val) {

printk(KERN_INFO "cannot assign the requested irq\n");

return -1;

}

else {

printk(KERN_INFO "Requested irq successful\n");

}

其中UART4_DMA_REG是DMA寄存器0x4a056000的基地址,请求的irq是45,sDMA中断的第1行。 在此UART寄存器初始化并启用DMA之后。 现在,用户调用write函数将100字节的数据复制到内核空间中的缓冲区。

下面的代码显示了写入功能:

ssize_t uart_write(struct file *filp,const char __user *buff, size_t count, loff_t *offp)

{

int no_of_bytes;

int maxbytes;

struct device *udevice = &devi;

int ret_mask;

char *kbuf = kmalloc(100,GFP_KERNEL|GFP_DMA);

maxbytes = BUFF_SIZE - *offp;

if(count > maxbytes)//overflow of buffer

no_of_bytes = maxbytes;

else

no_of_bytes = count;

if(no_of_bytes == 0)

printk(KERN_INFO "Nothing is there to write to device\n");

bytes_written = no_of_bytes - copy_from_user(kbuf,buff,no_of_bytes);//copy_from_user()returns remaining bytes.

printk(KERN_INFO "Write Completed\n");

Uindex = 0;

*offp += bytes_written;

ret_mask = dma_set_coherent_mask(udevice,DMA_BIT_MASK(32));

if(!ret_mask)

printk(KERN_INFO "set mask success \n");

else

printk(KERN_INFO "SET MASK NOT SUCCESS \n");

bus_addr = dma_map_single(udevice,kbuf,size,DMA_TO_DEVICE);

printk(KERN_INFO "dma_map_single completed");

dma_init();

return bytes_written;

}

dma_init(); 该功能初始化DMA寄存器,并在软件触发模式下启用通道。

void dma_init()

{

unsigned int ccr_val;

unsigned int csdp_val;

irq_line = 1; //for tx line 1 is considered

dma_cha_line = 0; //for tx line 0 is considered

/* Interrupt Enabled in DMA4_IRQENABLE_Lj and DMA4_CICRi registers */

iowrite32(0x1,(dma_map + 0x0018 + (4 * irq_line)));//to unmask the interrupt DMA4_IRQENABLE_Lj

iowrite32(0x8,(dma_map + 0x0088 + (0x60 * dma_cha_line)));//condition to generate interrupt CICR reg

/* Set the Read Port & Write Port access in CSDP */

csdp_val = ioread32(dma_map + 0x0090 + (0x60 * dma_cha_line));

csdp_val &= ~(0x3 << 7);//Source

csdp_val &= ~(0x3 << 14);//Destination

csdp_val &= ~(0x3 << 16);//Writing mode without posted

csdp_val &= ~(0x1 << 21);//little endian source

csdp_val &= ~(0x1 << 19);//little endian destination

csdp_val &= ~(0x1 << 13);//destination not packed

csdp_val &= ~(0x1 << 6);//source not packed

csdp_val &= ~(0x3);//ES is set to 8 bits

iowrite32(csdp_val,(dma_map + 0x0090 + (0x60 * dma_cha_line)));

/* CEN register configuration */

iowrite32(100,(dma_map + 0x0094 +(0x60 * dma_cha_line)));//EN is set to 1

/* CFN register configuration */

iowrite32(1,(dma_map + 0x0098 +(0x60 * dma_cha_line)));//FN is set to 1

/* Set the Channel Source & Destination start address */

iowrite32(bus_addr,(dma_map + 0x009C + (0x60 * dma_cha_line)));//Source

iowrite32(io_map,(dma_map + 0x00a0 + (0x60 * dma_cha_line)));//Destination

/* CCR configuration */

ccr_val = ioread32(dma_map + 0x0080 + (0x60 * dma_cha_line));

/* Set the Read Port & Write Port addressing mode in CCR */

/*

ccr_val &= ~(0x3 << 12);//Source - constant address mode

ccr_val |= (0x1 << 14);//Destination - post incremented address mode-set 14th bit and clear 15th bit

ccr_val &= ~(0x1 << 15);

*/

ccr_val |= (0x1 << 12);//source - post incremented address mode-set 12th bit and clear 13th bit

ccr_val &= ~(0x1 << 13);

ccr_val &= ~(0x3 << 14);//destination- constant address mode - clear 14 and 15th bit

ccr_val |= (0x1 << 26);//high priority on write

ccr_val &= ~(0x1 << 6);//low priority on read

ccr_val &= ~(0x1f);//CCR[4:0]

ccr_val &= ~(0x3 << 19);//CCR [19:20] to 0

ccr_val |= (0x1 << 7);// Set the channel enable bit in CCR

iowrite32(ccr_val,(dma_map + 0x0080 + (0x60 * dma_cha_line)));

/*CSEI,CSFI,CDEI,CDFI*/

iowrite32(1,(dma_map + 0x00a4 +(0x60 * dma_cha_line)));

iowrite32(1,(dma_map + 0x00a8 +(0x60 * dma_cha_line)));

iowrite32(1,(dma_map + 0x00ac +(0x60 * dma_cha_line)));

iowrite32(1,(dma_map + 0x00b0 +(0x60 * dma_cha_line)));

printk(KERN_INFO "DMA registers configured\n");

}

现在的问题是:一旦启用了通道(在调用dma_init()之后),就会调用ISR(处理程序)并进入无限循环。 我的ISR在写入模式下应该包含什么?

### 回答1: LinuxUART DMA模式是一种利用DMA(直接内存访问)技术来实现高效数据传输的模式DMA是一种数据传输方式,它可以直接在外设和内存之间进行数据传输,而无需CPU的干预。在UART通信中,传统的方式是通过CPU来处理接收和发送数据,而DMA模式则可以帮助减轻CPU的工作负担。 在Linux中,UART DMA模式的实现主要依赖于两个组件:串口(UART)控制器和DMA控制器。串口控制器负责与UART设备进行通信,而DMA控制器负责对DMA通道的管理和数据传输。在使用UART DMA模式时,首先需要配置串口控制器的相关参数,如波特率、数据位数等。然后,通过DMA控制器来配置DMA通道,指定数据的传输方向(接收或发送)以及源地址和目的地址。接下来,当有数据需要传输时,串口控制器将触发DMA控制器开始进行数据传输。DMA控制器将根据配置的参数进行数据的直接传输,并在传输完成后触发相应的中断,通知CPU来处理接收到的数据。 使用UART DMA模式的好处是可以提高数据传输的效率和吞吐量。由于数据传输的过程不需要CPU的直接参与,CPU可以专注于其他的任务,从而提高系统的整体性能。另外,使用DMA模式还可以减少数据传输过程中的延迟,提高系统的响应速度。 总而言之,LinuxUART DMA模式是一种通过利用DMA技术来实现高效数据传输的方式。它可以减轻CPU的负担,提高系统的整体性能,并提高数据传输的效率和吞吐量。 ### 回答2: Linux中的UART (通用异步收发传输器) 是通过DMA (直接内存访问) 模式来实现数据传输的一种方式。 在传统的UART模式中,数据的传输需要通过CPU来完成,即CPU需要处理数据的读取和写入操作。而在DMA模式下,数据的传输通过DMA控制器直接在外设和内存之间进行,而不需要CPU的干预。这样可以大大减轻CPU的负担,提高系统的响应速度和处理效率。 在Linux中,UARTDMA模式可以通过以下步骤来实现: 1. 配置UART控制器:首先需要配置UART控制器的寄存器,包括波特率、数据位数、停止位等参数。这可以通过修改相关的设备树节点或在设备驱动程序中进行。 2. 分配和初始化DMA通道:使用DMA引擎分配一个可用的DMA通道,并进行相关的初始化设置,如设置传输方向、内存地址等。 3. 配置DMA请求和传输:在UART控制器中,需要配置DMA请求和传输触发条件。这样当数据到达或离开UART缓冲区时,DMA控制器就会根据配置的条件触发相应的DMA传输。 4. 开始DMA传输:当DMA请求触发时,DMA控制器开始从UART缓冲区读取或写入数据,并将数据传输到指定的内存地址。 5. 处理DMA中断:当DMA传输完成时,DMA控制器会产生相应的中断信号,此时需要通过中断处理程序来处理DMA传输结果,如打印传输的数据或执行其他操作。 总结一下,Linux中的UART DMA模式可以通过配置UART控制器和DMA通道来实现数据的直接传输,无需CPU干预。这种模式可以提高系统的效率和响应速度,特别适用于高速数据传输和实时性要求较高的场景。 ### 回答3: UART (通用异步收发传输器)是一种常见的串口通信接口,广泛应用于嵌入式系统中。DMA (直接内存访问)是一种数据传输方式,可以将数据直接从外设(如UART)传输到内存,或从内存传输到外设,而无需CPU介入。Linux操作系统支持在UART通信中使用DMA模式。 使用LinuxUART DMA模式可以实现高效的数据传输。在传统的UART通信中,每个数据字节都需要经过CPU进行读取或写入操作,这会占用CPU的大量时间和计算资源。而使用DMA模式,可以将大量数据从UART接口直接传输到内存中,或将内存中的数据直接传输到UART接口中,无需CPU干预。这种方式可以极大地提高数据的传输速率和系统的效率。 在使用LinuxUART DMA模式时,需要配置DMA控制器和相应的DMA通道。通过配置DMA通道,可以指定数据的传输方向、传输大小和传输地址等参数。一旦配置完成,数据的传输将在DMA控制器的控制下自动进行,无需CPU干预。同时,通过使用DMA模式,还能够减少中断的频率,进一步提高系统的性能。 总结而言,LinuxUART DMA模式是一种通过配置DMA通道,实现高效数据传输的方式。它能够大幅度提高数据传输速率,减轻CPU的负担,提高系统的效率和性能。通过使用UART DMA模式,可以更好地满足嵌入式系统对数据传输的要求。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值