(新手)关于串口的编程的问题
怎么我一调,编译没有问题,进入板子执行就没有任何显现
当我把GPIO寄存器的那一段代码删除后,稍微改了下代码,就可以了
这是怎么回事呢?
GPIO不是没有调成控制串口的模式吗?
GPIO具有复用的功能 需要配置GPIO为GPIO模式还是串口模式
在使用S5PV210的串口发送和接收的时候,首先要对S5PV210的串口进行配置
1、配置GPIO,使对应管脚作为串口的发送和接收管脚
GPA0 0 1 管脚 2 3 可以配置
GPA0CON寄存器[7:4][3:0] 0x22
GPA0PUD寄存器[3:0] 0 禁止上下拉电阻
2、配置串口单元本身寄存器
ULCON0 0xE2900000
数据位:8位
停止位:1位
校验位:无
使用的正模式,非红外。
3、UCON0 0xE2900004
串口的收发模式:轮询
串口的时钟使用的PCLK UFCON0 0xE2900008
禁止FIFO
UMCON0 0xE290000C
禁止Modem
UBRdiv0 0xE2900028
UdivSLOT0 0xE290002C
UBRdiv0 = PCLK或者SCLK_UART/波特率/16 - 1 的整数部分
UdivSLOT0 查表,怎么查?
PCLK或者SCLK_UART/波特率/16 - 1 的小数部分 * 16 取整
查表
PCLK=66500000
波特率是115200
UBRdiv0 = 35
UdivSLOT0 = 0x0080
发送数据流程(轮询方式)
uart0_putc()
判断UTRSTAT0的BIT1,如果BIT1是0等待如果BIT1是1,就把要发送的一个字节数据写到发送寄存器(UTXH0,0xE2900020)
接收数据流程(轮询方式)
uart0_getc
判断UTRSTAT0的BIT0,如果BIT0是0等待如果BIT0是1,从URXH0 0xE2900024读取一个字节的数据。
编程时:
0xE2900000地址单元写3
0xE2900004地址单元写5
0xE2900008地址单元写0
0xE290000C地址单元写0
0xE2900028地址单元写35
0xE290002C地址单元写0x80
uart.h
- #ifndef _UART_H_
- #define _UART_H_
- #define GPA0CON (*(volatile unsigned int *)0xE0200000)
- #define GPA0PUD (*(volatile unsigned int *)0xE0200008)
- #define ULCON0 (*(volatile unsigned int *)0xE2900000)
- #define UCON0 (*(volatile unsigned int *)0xE2900004)
- #define UFCON0 (*(volatile unsigned int *)0xE2900008)
- #define UMCON0 (*(volatile unsigned int *)0xE290000C)
- #define UTRSTAT0 (*(volatile unsigned int *)0xE2900010)
- #define UTXH0 (*(volatile unsigned int *)0xE2900020)
- #define URXH0 (*(volatile unsigned int *)0xE2900024)
- #define UBRdiv0 (*(volatile unsigned int *)0xE2900028)
- #define UdivSLOT0 (*(volatile unsigned int *)0xE290002C)
- #define PCLK (66500000)
- //函数原型声明
- extern void uart0_init(void);
- extern void uart0_puts(const char *);
- extern void uart0_putc(char);
- extern char uart0_getc(void);
- extern void uart0_gets(char *,int);
- #endif // _UART_H_
uart.c
- #include "uart.h"
- //初始化串口寄存器
- void uart0_init(void){
- //配置GPIO口 根据CPU 手册中设置下面的寄存器
- //GPA0CON GPA0PUD
- ULCON0 = 3;
- UCON0 = 5;
- UFCON0 = 0;
- UMCON0 = 0;
- UBRdiv0 = 35;
- UdivSLOT0 = 0x0080;
- GPA0CON = 34;
- GPA0PUD = ~0xF;
- }
- //发送一个字符
- void uart0_putc(char c){
- //判断状态位
- while(!(UTRSTAT0 & (1<<1)));
- //发送字符
- UTXH0 = c;
- }
- //接收一个字符
- char uart0_getc(void){
- while(!(UTRSTAT0 & 1));
- return URXH0;
- }
- //接收一串字符
- void uart0_gets(char *str,int len){
- char* tmp = str;
- int in = len;
- //int i;
- while(--len){
- *tmp = uart0_getc();
- if(*tmp == '\r'){
- uart0_putc('\n'); //若此处为 \r 则不会输出,若为 \n 则在下一行跳跃输出字符的长度,然后输出字符串
- uart0_putc('\r');
- break;
- }
- if(*tmp == 127){ //127 是 ubuntu下 kermit软件中的 BACKSPACE按键 需要实现的效果就是当按下回车键的时候终端的上一个数据会被删掉,
- len++; //由于此分支的 127 输入到了 *tmp 中,此时的127是无用的,所以要进行 len++ ,但是有一个问题,我们的退格的目的是删除上一个字母,所以127的上一个字符也没用了,需要对len做两次自加进行还原 但是又出现一个问题,如果已经删到第0个元素就不能再自加两次了,这样会造成 len 越来越大。因此要在下面做一个判断
- if(len < in){
- len++;
- }
- if(tmp == str){
- continue;
- }
- uart0_putc('\b');
- uart0_putc(' ');
- uart0_putc('\b');
- --tmp;
- continue;
- }
- uart0_putc(*tmp);
- tmp++;
- }
- *tmp = 0;
- }
- //发送一串字符
- void uart0_puts(const char *str){
- if(str == 0){
- return;
- }
- while(*str){
- uart0_putc(*str);
- if(*str == '\n'){
- uart0_putc('\r');
- }
- str++;
- }
- }
谢谢小编分享!