005——串口移植(基于鸿蒙liteos-a)

目录

一、 Liteos-a中串口的使用

1.1 内核里打印

1.2 APP控制台

​编辑

1.2.1 /dev/console

1.2.2 /dev/serial

1.2.3 /dev/uartddev-0

1. 总体介绍

2. device_t

3. drvier_t

4. uartdev_fops

1.2.4 uart_ops

二、 鸿蒙串口内部的一些机制(流水账,新人可跳过)

三、串口移植

3.1 串口发送单个字符

3.2 在device_t中指定资源

3.3 实现uart_ops

3.4 GIC

四、启动后测试


一、 Liteos-a中串口的使用

1.1 内核里打印

内核打印函数是PRINT_RELEASE,它的内部调用关系如下:

PRINT_RELEASE
    LOS_LkPrint
        g_osLkHook
            OsLkDefaultFunc
                OsVprintf
                    UartPuts
                        UartPutsReg
                            UartPutStr
                                UartPutcReg

我们要实现UartPutcReg,用来输出单个字符。

(vscode的自动跳转太垃了,代码量大就失效了,这个图就不给大家截了。)

1.2 APP控制台

        我们编写的应用程序,调用printf时,那些信息从哪里打印出来?从控制台。 在串口上运行程序,控制台就是串口。 远程登录板子后运行程序,控制台就是远程登录终端。

控制台的实现分为4层:

        据老师所讲鸿蒙对每个文件都有个inode节点,里面有个 file_operation_vfs结构体。以open函数为例,最上层console会调用inode中的open,open会向下调用inode一层层最后到实际的uart设备inode的open。从而实现了设备分层。

代码系统初始化中就对console去做了初始化,有点类似于linux的bash

1.2.1 /dev/console

        init进程打开的就是/dev/console,它会打开shell。 我们在shell里执行各种APP时,这些APP会继承父进程的3个设备:标准输入、标准输出、标准错误,都对应/dev/console。 我们编写的APP,一般不需要自己去打开/dev/console,它已经继承得到了。 在串口上运行程序,/dev/console就是串口。 远程登录板子后运行程序,/dev/console就是远程登录终端。 所以/dev/console表示的是当前终端,它可能对应不同的设备,比如/dev/serial/dev/telnet

1.2.2 /dev/serial

        在Liteos-a中,/dev/serial被称为virtual serial,虚拟串口。它只是起一个中转的作用,无论是APP还是内核,使用/dev/serial时,都是再次跳转去执行具体串口设备驱动程序的函数。比如:

        那么,/dev/serial这个虚拟串口,怎么跟具体串口挂钩?也就是上图中,GetFileOps函数为何能得到具体串口的驱动程序? 方法如下图所示:

virtual_serial_init函数会找到/dev/uartdev-0的驱动程序(即它对应的struct inode,里面含有file_operations_vfs)。

1.2.3 /dev/uartddev-0

1. 总体介绍

这是真正操作硬件的驱动程序,它分为两部分:device_tdriver_t

  • device_t中设置资源,比如寄存器物理基地址、中断号等

  • driver_t中提供函数,比如device_probedevice_attach函数

    • 当内核发现有名字系统的device_tdriver_t

    • 就会调用driver_t中的device_probedevice_attach函数

    • 在里面根据device_t得到资源、注册驱动register_driver

这种写驱动程序的方法,被称为分离:操作函数、资源分离。 以后想换一个硬件,只需要修改device_t就可以,driver_t保存不变。

2. device_t

示例代码:

3. drvier_t

先注册一个drvier_t结构体,它里面带有各类device_method_t

当内核发现有同名的device_tdriver_t时,就会调用driver_t里面提供的device_probedevice_attach函数。

4. uartdev_fops

device_attach函数里从device_t里获取硬件资源、注册驱动:

/dev/uartdev-0对应的驱动程序时uartdev_fops,它通过uart_ops来操作硬件。

1.2.4 uart_ops

在UART驱动程序里,我们只需要提供硬件操作部分:

uart_ops里有4个函数:

  • config:配置串口,比如波特率等

  • startup:启动串口,比如注册中断处理函数、启动串口

  • start_tx:发送字符串

  • shutdown:关闭串口

串口就两大功能:发送数据、接收数据。 在Liteos-a中,发送数据比较简单:没有使用中断,而是使用查询方式逐个发送,核心是UartPutcReg。 接收数据时使用中断,所以需要注册串口接收中断处理函数,它要做的事情是:

  • 发生中断时,读取硬件获得字符,可能有多个字符

  • 处理特殊字符:比如不`\r`换为`\n`

  • 通知上层代码:udd->recv(udd, buf, count);

 

二、 鸿蒙串口内部的一些机制(流水账,新人可跳过)

        鸿蒙和其它操作系统一样,将设备分成了driver和device。device用来描述设备信息。driver用来做一些操作。比如不同品牌的内存我要用它来做的事情都是一样的,这部分的代码就是driver,而具体的它每个功能寄存器的地址,中断等等则在device里。

        这是一个设备的结构体对象,它的执行顺序其实就是从上往下的,先probe在attach(获得内存资源和中断资源)在detach最后shutdown结束。

        这里是它的一个调用过程,韦东山老师讲的这个课不太适合初学者,适合有一定基础的人。天才除外。我指的是正常人哦。因为我之前对linux的学习比较多加上去过好多公司实习,对底层的东西比较了解。鸿蒙的思想和linux很像所以可以快速上手。随着不断的学习,我发现大佬们的思想是一致的。很多优秀的项目都有共同的内核,这里不是操作系统的那个内核哦,是思想。最近也遇到了很多的问题,还是那句话,感兴趣,有精力的同学可以加群一起学,最好是我们一人研究一块后面在交流分享,这其实就是团队学习的意义。可以快速高效的掌握新技术。主要是我赶时间,呜呜呜呜。

        回到正题我们想要移植一个硬件设备其实就完成那个结构体就行,比较简单,其它的东西内核已经帮我们做了。linux驱动开发时也是这样,完成设备结构体中需要的函数成员,把结构体一写就好了,不论简单的设备还是复杂的设备思想上都是这样的。

三、串口移植

我们的目标是:让最小系统启动。 那么对于串口,不需要考虑得很全面:

  • 不需要初始化串口:u-boot已经初始化串口了

  • 不需要动态配置串口:固定使用某个波特率等配置就可以(在u-boot里设置过了)

移植工作只需要实现这几点:

  • 串口发送单个字符

  • 注册串口接收中断函数:确定中断号、使能中断、在中断函数中读取数据

3.1 串口发送单个字符

首先改下相关代码的名字

功能代码已经实现了,我们就改改地址啥的就行。

3.2 在device_t中指定资源

需要确定2个资源:寄存器地址、中断号

 这里设置串口的物理基地址

 

这里是串口的中断号

这里科普一下这里的SPI可不是那个通信总线哦。 

        SPI,即Shared Peripheral Interrupts,代表共享外设中断。这种中断类型来自于多个外设设备,例如IO外设等,并且这些设备共享相同的中断线。当任何一个这些外设需要引起CPU的注意时,它们会触发SPI。由于多个设备共享同一中断线,因此中断处理程序需要能够区分是哪一个设备触发了中断,这通常通过读取中断状态寄存器来实现。

        SGI,即Software Generated Interrupts,是软件生成的中断。这种中断不是由硬件外设触发的,而是由软件程序(如操作系统或驱动程序)通过写入特定的寄存器或执行特定的指令来生成的。SGI通常用于软件层面的通信和事件处理,例如任务调度、线程同步等。由于SGI是由软件控制的,因此它们具有高度的灵活性和可编程性。

        PPI,通常指CPU Private Peripheral Interrupts,即CPU私有外设中断。这是一种中断类型,用于处理CPU与私有外设之间的通信和事件响应。PPI在计算机系统中扮演着重要角色,帮助CPU及时响应和处理来自外设的中断请求。

然后

我们改成串口2,因为exynos4412的uboot初始化的是串口2的资源

然后这里也有一个坑这里需要修改 

        还有这里不是改PBASE,PBASE都没人用,改BASE或者说PBASE是base的地址,需要一个宏转化一下

想找下这个宏在哪定义的这个sg vscode恶心死了,跳转一直加载不出来,可恶。换source insight

访问虚拟机的IP然后映射到本地网络虚拟磁盘方便操作

 新建个目录放工程文件,这里没有权限。

sudo chmod -R go+rwx /home/book

现在就没问题了

新建工程

添加vendor目录

添加kernel里的liteos-a目录

添加drivers目录

这里添加汇编文件不然会加不进来

点开后把liteos-a重新加一下,然后同步文件

有点慢等一会,其它操作可以参考我以前的文章

Source Insight的学习_source insight read only-CSDN博客

不知道为什么要这样,猜测不能用物理地址可能要转化成对应虚拟地址用

这些名字也需要改

然后里面的内容做个替换

 然后我们来编译一下

        整型溢出了 

        发现鸿蒙源码的一个bug这里为什么用有符号整型啊,我基地址0x4000 0000,内存大小一个G加起来是0x8000 0000改好超过有符号整型的限制。内存映射限制在有符号整型的话岂不是意味着设备最多只能有2个G的内存。

        我在官方的社区下发了帖子不知道会不会回我。下面这个链接。有人回的话兄弟们踢我一下。

https://bbs.csdn.net/topics/618273814

这里先改小一点点。

又来咯

再给LCD让一部分空间,无语啦。

 

 

3.3 实现uart_ops

在UART驱动程序里,uart_ops结构体封装了UART的硬件操作:

uart_ops里有4个函数:

  • config:配置串口,比如波特率等

  • startup:启动串口,比如注册中断处理函数、启动串口

  • start_tx:发送字符串

  • shutdown:关闭串口

我们只需要实现startup、start_tx,其他函数可以设为空:

  • startup:确定中断号、request_irq、使能中断、提供中断处理函数

  • start_tx:发送字符串

3.4 GIC

kernel\liteos_a\platform\main.c中,调用OsSystemInfo打印系统信息时,代码如下:

    PRINT_RELEASE("\n******************Welcome******************\n\n"
            "Processor   : %s"
#if (LOSCFG_KERNEL_SMP == YES)
            " * %d\n"
            "Run Mode    : SMP\n"
#else
            "\n"
            "Run Mode    : UP\n"
#endif
            "GIC Rev     : %s\n"
            "build time  : %s %s\n"
            "Kernel      : %s %d.%d.%d.%d/%s\n"
            "\n*******************************************\n",
            LOS_CpuInfo(),
#if (LOSCFG_KERNEL_SMP == YES)
            LOSCFG_KERNEL_SMP_CORE_NUM,
#endif
            HalIrqVersion(), __DATE__, __TIME__,\
            KERNEL_NAME, KERNEL_MAJOR, KERNEL_MINOR, KERNEL_PATCH, KERNEL_ITRE, buildType);

里面的HalIrqVersion函数用到的GIC的虚拟地址,要正确设置,否则没有打印信息。 IMX6ULL的内存映射代码里,设备空间从GIC开始映射,所以GIC的虚拟地址就是PERIPH_DEVICE_BASE

// kernel/liteos_a/kernel/base/include/los_vm_zone.h
#define GIC_VIRT_BASE    PERIPH_DEVICE_BASE
​
// vendor/democom/demochip/board/include/asm/platform.h
#define GIC_BASE_ADDR             (GIC_VIRT_BASE)

四、启动后测试

        在启动文件中有这样一段

这样启动时串口有打印我们就知道启动成功了

栈的初始化我们是从0x4000 0000开始的后面的大小是内核的大小liteos-a不会大于16MB

        看过我之前做linux系统移植的兄弟们应该知道之前在4412上运行裸机程序都是在40008000这个地址上,我猜测这个8000可能是uboot的大小,但是后面问了华清的老师这个是武老师的个人喜好,好家伙我直接好家伙。uboot其实不在这运行,在内存运行的话不久死循环了么,因为内存是uboot初始化的哇哈哈。应该是soc内部还有一小段空间可供uboot在启动时使用,或者可能cpu直接去存放uboot的ROM里取的,以上都是猜测后面有机会研究一下这个芯片的启动过程。

        汇编要跳转的测试函数我们放到这个文件里,这里要定义一下串口控制器的位置,其实这里有个问题,就是这个代码不是通用的我们这个等待发送完成要根据自己的芯片手册做偏移。还有这个发送缓冲区的位置,这个通常也是个寄存器地址和这个串口相关的代码大概都在串口的地址做偏移的位置,一般会对整个串口的寄存器做个大的结构体。这里由于架构就不对我们只能先意思一下咯。

虚拟地址加载完在做个打印测试

这里我们前面定义过宏,切记写代码不要魔鬼数字,后面自己都不知道是干嘛的了。

因为对应结构体没实现后面这里一定会报错的,我就先注释掉了

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/479802.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

阿里云4核8G服务器优惠价格表,ECS u1实例

阿里云4核8G服务器优惠价格955元一年,配置为ECS通用算力型u1实例(ecs.u1-c1m2.xlarge)4核8G配置、1M到3M带宽可选、ESSD Entry系统盘20G到40G可选,CPU采用Intel(R) Xeon(R) Platinum处理器,阿里云活动链接 aliyunfuwuq…

就业班 第二阶段 2401--3.21 day3 备份

一、逻辑备份 备份的是建表、建库、插入等操作所执行SQL语句,适用于中小型数据库,效率相对较低。 本质:导出的是SQL语句文件 优点:不论是什么存储引擎,都可以用mysqldump备成SQL语句 缺点:速度较慢&…

谷歌推出通用AI代理:能自动执行600多种动作,游玩复杂3D游戏

谷歌DeepMind的研究人员推出了一种面向3D环境的通用AI代理——SIMA。 SIMA无需访问游戏的源代码,也不需要定制的API。只需要输入图像和用户提供的简单自然语言文本指令,SIMA就能像人类玩家一样执行走路、跑步、建造、打开地图等各种游戏中的操作。 为了…

市场复盘总结 20240322

仅用于记录当天的市场情况,用于统计交易策略的适用情况,以便程序回测 短线核心:不参与任何级别的调整,采用龙空龙模式 一支股票 10%的时候可以操作, 90%的时间适合空仓等待 二进三: 进级率中 36% 最常用…

C++初阶:vector的使用与自实现

目录 1. vector与顺序表2. vector的使用2.1 默认成员函数2.2 vector的迭代器2.3 容量相关成员函数2.4 遍历访问方式2.5 vector的修改操作 3. vector的自实现3.1 自实现功能3.2 vector的结构与自实现的逻辑3.3 自实现vector功能接口 1. vector与顺序表 我们在初阶数据结构中学习…

34 | 到底可不可以使用join?

在实际生产中,关于 join 语句使用的问题,一般会集中在以下两类: 1. 我们 DBA 不让使用 join,使用 join 有什么问题呢? 2. 如果有两个大小不同的表做 join,应该用哪个表做驱动表呢? 今天这篇文…

【进程和线程】操作系统中的并发执行机制

目录 一、什么是进程(Process)? 进程的管理 进程调度(重点) 二、什么是线程(Thread)? 三、进程和线程的区别与联系 进程(Process) 线程(Thread) 总结比较 一、什么是进程(Process)? 进程和线程是操作系统中一个非常核心的话题&#…

鸿蒙Harmony应用开发—ArkTS-@Provide装饰器和@Consume装饰器:与后代组件双向同步

Provide和Consume,应用于与后代组件的双向数据同步,应用于状态数据在多个层级之间传递的场景。不同于上文提到的父子组件之间通过命名参数机制传递,Provide和Consume摆脱参数传递机制的束缚,实现跨层级传递。 其中Provide装饰的变…

随笔】Git -- 常用命令(四)

💌 所属专栏:【Git】 😀 作  者:我是夜阑的狗🐶 🚀 个人简介:一个正在努力学技术的CV工程师,专注基础和实战分享 ,欢迎咨询! 💖 欢迎大…

OpenHarmony实现一次开发多端部署分布式新闻客户端页面

分布式新闻客户端(ArkTS) 介绍 本篇Codelab基于栅格布局、设备管理和多端协同,实现一次开发,多端部署的分布式新闻客户端页面。主要包含以下功能: 展示新闻列表以及左右滑动切换新闻Tab。点击新闻展示新闻详情页。点…

产生死锁的四大条件

死锁 由于两个或两个以上的线程相互争夺对方的资源,而同时不释放自己的资源,导致所有线程同时被阻塞 产生死锁的四大条件 互斥条件:一个资源在同一时刻只能由一个运算单元(进程、线程或协程)占用(排它性&…

停车场引导与道闸系统工程解析

停车场引导和道闸系统工程是一个综合性的智能车辆管理系统,它结合了现代电子信息技术,旨在实现安全、高效、自动化的停车管理。以下是该系统的主要组成部分及其功能: 1. 车牌识别技术:这是现代停车场管理的核心,能够自…

华清远见作业第五十三天——ARM(第七天)

代码 key_inc.h #ifndef __KEY_INC_H__ #define __KEY_INC_H__ #include "stm32mp1xx_gic.h" #include "stm32mp1xx_exti.h" #include "stm32mp1xx_rcc.h" #include "stm32mp1xx_gpio.h"void key1_it_config();void key2_it_config(…

C++概述

目录 一、C关键字(63个) 二、C几个关键点: 三、C语言缺陷一:命名冲突 四、C新概念:命名空间(namespace) 五、命名空间的嵌套: 六、展开命名空间:(using …

【收藏】什么是API测试?这是我见过的最全的测试指南!

在最近的部署中,当我被问到“什么是API测试?”时,我正与客户一起制定API测试策略。那时我突然意识到,要描述API测试居然是一件如此具有挑战性的事情,即使你如实地描述了它,也往往听起来很无聊和复杂。 好吧…

FloodFill算法——岛屿数量

文章目录 题目解析算法解析代码解析 题目解析 岛屿数量 题目依旧是熟悉的配方,熟悉的味道,还是那个0还是那个1还是那个二维矩阵,这时候BFS和DFS闻着味就来了,我们来看一下这个题目,这个题目也很容易理解如下图有一个…

【No.14】蓝桥杯贪心法|最少硬币问题|活动安排问题(4)|翻硬币|快乐司机|防御力|答疑(C++)

算法优点 容易理解:生活常见 操作简单:在每一步都选局部最优 效率高:复杂度常常是O(1)的 算法缺点 局部最优不一定是全局最优 贪心算法(Greedy algorithm),又称贪婪算法。是一种在每一步选择中都采取在…

教你在PC客户端中集成镭速高速传输插件

企业一直以来对文件传输的速度和安全性都有着严苛的要求,传统的FTP/HTTP传输方式因速度慢、易受网络延迟影响、数据包丢失等问题而不再适应现代企业的需求。镭速高速传输插件应运而生,为企业提供了一个快速、安全的文件传输新选择。本文将详细介绍如何在…

代码随想录算法训练营第day54|392.判断子序列 、 115.不同的子序列

目录 392.判断子序列 115.不同的子序列 392.判断子序列 力扣题目链接(opens new window) 给定字符串 s 和 t ,判断 s 是否为 t 的子序列。 字符串的一个子序列是原始字符串删除一些(也可以不删除)字符而不改变剩余字符相对位置形成的新字…

【WEEK4】 【DAY3】整合SSM框架之功能实现—修改、删除数据【中文版】

2024.3.20 Wednesday 接上文【WEEK4】 【DAY2】整合SSM框架之功能实现—总览、添加数据【中文版】 目录 7.6.修改功能7.6.1.修改BookController.java7.6.2.修改allBook.jsp7.6.3.新建updateBook.jsp7.6.4.修改MyBatis-config.xml7.6.5.运行 7.7.删除功能7.7.1.修改BookContro…