天天看點

Hi3519v101 uart驅動

一、說明

1、Hi3519v101 SDK已經将uart驅動寫好了,可以直接使用。但核心預設隻添加了uart0驅動,其他端口需要手動添加。uart驅動路徑為:

drivers/tty/serial/amba-pl011.c
           

2、采用SDK的根檔案系統啟動後,可以在dev檔案夾下面看到ttyAMA0、ttyAMA1。這是根檔案系統添加的兩個裝置,但ttyAMA1并沒有驅動,運用程式通過open函數是不能打開的。

/ # ls dev/ttyAMA*
dev/ttyAMA0  dev/ttyAMA1
/ #
           

根檔案添加序列槽裝置檔案路徑:

cat /home/osrc/Hi3519V101_SDK_V1.0.4.0/package/image_big-little/rootfs_uclibc_big-little/etc/init.d/S00devs
#!/bin/sh

mknod /dev/console c 5 1
mknod /dev/ttyAMA0 c 204 64
mknod /dev/ttyAMA1 c 204 65
mknod /dev/ttyS000 c 204 64
mknod /dev/null    c 1 3
           

檢視驅動狀态,隻有uart0

/ # cat proc/tty/driver/ttyAMA
serinfo:1.0 driver revision:
0: uart:PL011 rev2 mmio:0x12100000 irq:36 tx:4976 rx:40 RTS|CTS|DTR|DSR|CD|RI
           

核心啟動列印資訊

Serial: AMBA PL011 UART driver
12100000.uart: ttyAMA0 at MMIO 0x12100000 (irq = 36, base_baud = 0) is a PL011 rev2
console [ttyAMA0] enabled
           

二、添加uart1驅動

未解決uart1無法打開問題,核心需要做如下修改

1、打開arch/arm/boot/dts/hisi-hi3519v101.dtsi

aliases {
		serial0 = &uart0;
		serial1 = &uart1;//新增
		i2c0 = &i2c_bus0;
           

2、打開arch/arm/boot/dts/hisi-hi3519v101-hmp-demb.dts

&uart0 {
	status = "okay";
};

//新增
&uart1 {
	status = "okay";
};
           

3、打開arch/arm/boot/dts/hisi-hi3519v101-demb.dts

&uart0 {
	status = "okay";
};

//新增
&uart1 {
	status = "okay";
};
           

4、編譯核心

make ARCH=arm CROSS_COMPILE=arm-hisiv500-linux- clean
make ARCH=arm CROSS_COMPILE=arm-hisiv500-linux- uImage
           

5、下載下傳新核心,測試

核心啟動資訊

Serial: AMBA PL011 UART driver
12100000.uart: ttyAMA0 at MMIO 0x12100000 (irq = 36, base_baud = 0) is a PL011 rev2
console [ttyAMA0] enabled
12101000.uart: ttyAMA1 at MMIO 0x12101000 (irq = 37, base_baud = 0) is a PL011 rev2
SCSI subsystem initialized
           

驅動狀态

# cat proc/tty/driver/ttyAMA
serinfo:1.0 driver revision:
0: uart:PL011 rev2 mmio:0x12100000 irq:36 tx:4976 rx:40 RTS|CTS|DTR|DSR|CD|RI
1: uart:PL011 rev2 mmio:0x12101000 irq:37 tx:0 rx:0 DSR|CD|RI
           

6、編寫驅動檔案、運用程式

uart/uart_drv.c  這裡隻做io複用配置

#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/delay.h>
#include <linux/ide.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/errno.h>
#include <linux/gpio.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/of_gpio.h>
#include <linux/semaphore.h>
#include <linux/timer.h>
#include <linux/i2c.h>
#include <asm/mach/map.h>
#include <asm/uaccess.h>
#include <asm/io.h>

//spi gpio管腳複用實體位址 set 0x01 = spi mode
#define spi2_cs0_base 0x12040058
#define spi2_sclk_base 0x1204004c
#define spi2_sdo_base 0x12040050
#define spi2_sdi_base 0x12040054

//uart1 gpio管腳複用實體位址 set 0x01 = uart mode
#define uart1_rxd_base 0x1204008c
#define uart1_txd_base 0x12040094


#define gpio_phy_data 0x010 //gpio管腳資料偏移位址,0b00_0001_0000
#define gpio_phy_dir 0x400  //管腳方向寄存器偏移位址
#define gpio_out 0x4        //gpio3_2 配置為輸出,1輸出 0輸入def
#define gpio_out_h 0xFF
#define gpio_out_l 0x00
#define spi2_mode 0x01
#define uart_mode 0x01

static void __iomem *spi2_cs0;
static void __iomem *spi2_sclk;
static void __iomem *spi2_sdo;
static void __iomem *spi2_sdi;


static void __iomem *uart_rxd;
static void __iomem *uart_txd;



static int __init uart_drv_init(void)
{
    //管腳複用配置  4個位元組,32位
    uart_rxd = ioremap(uart1_rxd_base, 4);
    uart_txd = ioremap(uart1_txd_base, 4);

    iowrite32(uart_mode, uart_rxd);
    iowrite32(uart_mode, uart_txd);

    return 0;
}

static void __exit uart_drv_exit(void)
{
    iounmap(uart_rxd);
    iounmap(uart_txd);
}

module_init(uart_drv_init);
module_exit(uart_drv_exit);
MODULE_LICENSE("GPL");
           

uart/test_uart.c 這裡沒做序列槽參數配置,也可參考大佬的一篇文章《海思HI35xx平台序列槽配置方法》

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <sys/time.h>

int main(int argc, char **argv)
{
	int fd;
	unsigned char val;
 
	fd = open("/dev/ttyAMA1", O_RDWR);	//打開裝置
 
    if(fd < 0)
	{
		printf("can`t open!\n");
		return 0;	
	}
        

	if(argc != 2)
	{
		printf("Usage :\n");
		printf("%s <read|write>\n", argv[0]);
		return 0;
	}

	if (strcmp(argv[1], "read") == 0)
	{
		read(fd, &val, 1);
	}
	else
	{
		val = 0x55;
		write(fd, &val, 1);
		printf("write 0x%x\n", val);
	}
	
	return 0;
}
           

用示波器測試uart_txd管腳波形

Hi3519v101 uart驅動

繼續閱讀