S3C2440串口的基本使用

发布时间:2023-09-25  

2440A有三个串口,我们使用串口0对它进行了解熟悉。

首先肯定是应该找到手册上串口0所对应的引脚,然后配置相应寄存器。

串口0对应GPIO H的 2,3

串口在单片机中我们已经有很多使用经验了,对于协议采用 8-N-1,8bit数据位,无校验,1停止位。

说明波特率的计算方式:

 

把串口对应IO配置成 TX和RX功能之后,我们需要对指定寄存器进行读写操作,实现串口的接发。

具体的寄存器就不贴出来了。手册上都有,这里不使用FIFO和中断方式,只是最基本的接发操作。


main.c:

#include "s3c2440_gpio.h"

#include "s3c2440_soc.h"

#include "uart.h"


void SystemInit(void)

{

    //配置LOCKTIME(0x4C000000) = 0xFFFFFFFF 

    *(volatile unsigned int *)0x4C000000=0xFFFFFFFF;

    //CLKDIVN(0x4C000014) = 0X5, tFCLK:tHCLK:tPCLK = 1:4:8

    *(volatile unsigned int *)0x4C000014=0x5;

    //协处理指令

    __asm__(

    "mrc    p15, 0, r1, c1, c0, 0n"        /* 读出控制寄存器 */ 

    "orr    r1, r1, #0xc0000000n"          /* 设置为“asynchronous bus mode” */

    "mcr    p15, 0, r1, c1, c0, 0n"        /* 写入控制寄存器 */

    );

    /* 设置MPLLCON(0x4C000004) = (92<<12)|(1<<4)|(1<<0) 

     *  m = MDIV+8 = 92+8=100

     *  p = PDIV+2 = 1+2 = 3

     *  s = SDIV = 1

     *  FCLK = 2*m*Fin/(p*2^s) = 2*100*12/(3*2^1)=400M

     */

    *(volatile unsigned int *)0x4C000004=(92<<12)|(1<<4)|(1<<0);

}

void Delay(uint32_t count)

{

    while(count--);

}

int main(void)

{

    //配置GPIOF 0,2,GPIOG 3为输入模式

    Set_gpio(IN, GPIOF,GPIO_PinSource0);

    Set_gpio(IN, GPIOF,GPIO_PinSource2);

    Set_gpio(IN, GPIOG,GPIO_PinSource3);

    //点亮LED1然后熄灭

    Reset_gpio(OUT, GPIOF,GPIO_PinSource4);

    Delay(100000);

    Set_gpio(OUT, GPIOF,GPIO_PinSource4);

    Delay(100000);

    //点亮LED2然后熄灭

    Reset_gpio(OUT, GPIOF,GPIO_PinSource5);

    Delay(100000);

    Set_gpio(OUT, GPIOF,GPIO_PinSource5);

    Delay(100000);

    //点亮LED3然后熄灭

    Reset_gpio(OUT, GPIOF,GPIO_PinSource6);

    Delay(100000);

    Set_gpio(OUT, GPIOF,GPIO_PinSource6);

    Delay(100000);

    //熄灭三盏LED灯

    Set_gpio(OUT, GPIOF,GPIO_PinSource5);

    Set_gpio(OUT, GPIOF,GPIO_PinSource4);


    unsigned char c;

    

    uart_init();

    puts("Hello, world!nr");


    while (1)

    {

        

        while(1)

        {

            c = getchar();

            if (c == 'r')

            {

                putchar('n');

            }

        

            if (c == 'n')

            {

                putchar('r');

            }

        

            putchar(c);

        }



    }

    return 0;

}


uart.c:


#include "uart.h"

#include "s3c2440_soc.h"

#define PCLK            50000000    // PCLK为50MHz

#define UART_CLK        PCLK        //  UART0的时钟源设为PCLK

#define UART_BAUD_RATE  115200      // 波特率

#define UART_BRD        ((int)(UART_CLK  / (UART_BAUD_RATE * 16.0)+0.5) - 1)

#define NULL 0

void uart_init(void)

{

    /* GPH2,3用于TxD0, RxD0 */

    //清除GPHCON

    GPHCON &= ~((3<<4) | (3<<6));

    //设置2,3为发送接收

    GPHCON |= ((2<<4) | (2<<6));

    //使能内部上拉 

    GPHUP &= ~((1<<2) | (1<<3));  

    ULCON0  = 0x03;     // 8N1(8个数据位,无校验,1个停止位)

    UCON0   = 0x05;     // 中断/查询模式,UART时钟源为PCLK

    UFCON0  = 0x00;     // 不使用FIFO

    UMCON0  = 0x00;     // 不使用流控

    UBRDIV0 = UART_BRD; // 波特率为115200


}


int putchar(int c)

{

    /* UTRSTAT0 */

    /* UTXH0 */

    /*发送是向寄存器写数据*/

    while (!(UTRSTAT0 & (1<<2)));

    UTXH0 = (unsigned char)c;

    return 0;

    

}


int getchar(void)

{

//接收是在寄存器中读数据

    while (!(UTRSTAT0 & (1<<0)));

    return URXH0;

}


int puts(const char *s)

{

    if(s!=NULL)

    {

        while (*s)

        {

            putchar(*s);

            s++;

        }

    }

    return 0;

}


配置串口,要注意波特率,还有是否有缓存,实际应用中,FIFO是不可缺少的,这里只是入门简单实现。


启动文件和之前的时钟章节一样,


Makefile:


all:

    arm-linux-gcc -c -g -o s3c2440_gpio.o s3c2440_gpio.c

    arm-linux-gcc -c -g -o uart.o uart.c

    arm-linux-gcc -c -g -o main.o main.c

    arm-linux-gcc -c -g -o start.o start.S

    arm-linux-ld -Ttext 0 start.o main.o s3c2440_gpio.o uart.o -o uart.elf

    arm-linux-objcopy -O binary -S uart.elf uart.bin

    arm-linux-objdump -S -D uart.elf > uart.dis

clean:

    rm *.bin *.o *.elf *.dis


NOTE:


在此之后,还是不去编写库函数了,太过于消耗时间,使用寄存器配置的方式,如果工作中需要长期使用某个cpu或mcu时,再去编写对应的库函数,因为时间实在不够。


文章来源于:电子工程世界    原文链接
本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。

相关文章

    -A Standard)是Apple的Macintosh计算机的串口连接标准。RS-422使用差分信号,RS-232使用非平衡参考地的信号。差分传输使用两根线发送和接收信号,对比RS-232,它能......
    行通信设备。那么,它与UART有什么区别呢?它的硬件连接方式是怎样的?该如何使能USART的SPI模式?把USART当做SPI来使用的时候和标准的SPI有什么区别,需要注意哪些事项?本文......
    规定接口插件电缆以及使用的协议。   2、串口通信协议   在串口通信中,常用的协议包括RS-232、RS-422RS-485。   •RS-232:标准串口,最常用的一种串行通讯接口。有三种类型(A,BC......
    接收和发送指示状态灯,此外用了一个叫MAX3232 的芯片,那它是用来实现什么的呢?首先我们要知道计算机上的串口是具有RS-232 标准的串行接口,而RS-232 的标准中定义了其电气特性:高电平“1”信号......
    什么是人机界面?人机界面跟触摸屏有什么区别?;人机界面与人们常说的“触摸屏”有什么区别?从严格意义上来说,两者是有本质上的区别的。因为“触摸屏”仅是人机界面产品中可能用到的硬件部分,是一......
    什么是串口通讯?STM32中的串口通讯接口有哪些;所谓通信就是两种设备之间进行数据交互,从而传递大料的信息,因为通讯的存在,使得大量的数据有了传输通道,各种信息得到了汇总,方便了人们的生活 。MCU......
    是很清楚的朋友要注意看看了哦,在最后还会为大家分享有些关于stm32的视频资料便于学习参考。   什么是串口   UART : Universal Asynchronous Receiver......
    stm32学习心得(2022-12-16)
    后还会为大家分享有些关于stm32的视频资料便于学习参考。   什么是串口   UART : Universal Asynchronous Receiver/Transmitter 通用......
    后还会为大家分享有些关于stm32的视频资料便于学习参考。 什么是串口 UART : Universal Asynchronous Receiver/Transmitter 通用异步收发器 USART......
    行通信接口 常见的串行通信接口 STM32串口通信基础 STM32的串口通信接口有两种,分别是:UART(通用异步收发器)、USART(通用同步异步收发器)。而对于大容量STM32F10x系列......

我们与500+贴片厂合作,完美满足客户的定制需求。为品牌提供定制化的推广方案、专属产品特色页,多渠道推广,SEM/SEO精准营销以及与公众号的联合推广...详细>>

利用葫芦芯平台的卓越技术服务和新产品推广能力,原厂代理能轻松打入消费物联网(IOT)、信息与通信(ICT)、汽车及新能源汽车、工业自动化及工业物联网、装备及功率电子...详细>>

充分利用其强大的电子元器件采购流量,创新性地为这些物料提供了一个全新的窗口。我们的高效数字营销技术,不仅可以助你轻松识别与连接到需求方,更能够极大地提高“闲置物料”的处理能力,通过葫芦芯平台...详细>>

我们的目标很明确:构建一个全方位的半导体产业生态系统。成为一家全球领先的半导体互联网生态公司。目前,我们已成功打造了智能汽车、智能家居、大健康医疗、机器人和材料等五大生态领域。更为重要的是...详细>>

我们深知加工与定制类服务商的价值和重要性,因此,我们倾力为您提供最顶尖的营销资源。在我们的平台上,您可以直接接触到100万的研发工程师和采购工程师,以及10万的活跃客户群体...详细>>

凭借我们强大的专业流量和尖端的互联网数字营销技术,我们承诺为原厂提供免费的产品资料推广服务。无论是最新的资讯、技术动态还是创新产品,都可以通过我们的平台迅速传达给目标客户...详细>>

我们不止于将线索转化为潜在客户。葫芦芯平台致力于形成业务闭环,从引流、宣传到最终销售,全程跟进,确保每一个potential lead都得到妥善处理,从而大幅提高转化率。不仅如此...详细>>