海思芯片上GPIO和PWM操作

2023-11-02

一、GPIO的配置

    GPIO的设置一般为三步:1 设置gpio端口复用、2 设置GPIO口的方向、3 读取或者写入GPIO值;第一步不是每个GPIO口都是要配置的,如果你设置的GPIO端口有复用功能,那么你需要对GPIO对应复用寄存器进行配置,例如:文档中GPIO5_5是有复用功能的,0:PWM_OUT0  ; 1:GPIO5_5,所以把复用寄存器设置为1就是表示为GPIO5_5;但有些GPIO端口是没有复用功能的,例如GPIO0_0,这个就不需要复用寄存器的设置;

1、设置gpio端口复用

     所谓复用,就是指一个端口能同时干很多事,但是具体你要求他干啥,是需要设置的,这就是管脚复用寄存器的作用。
举个例子,如果我们想设置GPIO5_5,需要首先找到对应的管脚复用寄存器地址;通过手册查询,如果你想让这个管脚作为io口使用,需要给复用控制寄存器muxctrl_reg54这个寄存器里写0x01;管脚复用寄存器的基地址为0x120f_0000,muxctrl_reg54的偏移地址0xD8,所以这个寄存器的地址为0x120f00D8;

通过HIS自带的himm命令输入以下即可:
himm 0x120F00D8 0x01;

     这样GPIO5_5对应管脚的功能就设置为了IO口,还有一个问题需要注意:在设置寄存器的时候要查看该寄存器其他bit位的作用,不要盲目的直接设置你需要设置的bit位,可能会其他bit给清除了,所以最好的做法是先读出寄存器的值,然后或上设置的位的值,再设置寄存器,这样就不会改变其他bit位的值;

                                     

 

                                  

2 设置GPIO口的方向

    所谓方向,就是你是把io做输入用还是输出用。比如如果你的板子外接了个电池模块,你希望电池模块在没电的时候,能够给你的板子一个信号,告诉你没电了,这时你就需要将io口作为输入使用。反过来,如果你需要告诉外接的什么设备什么事情,那就要将io口做输出,设置GPIO口的方向,实际上就是写 GPIO_DIR寄存器。同样的,首先需要找地址,对于GPIO5的基地址为:0x121A_0000;GPIO_DIR的偏移地址为0x400,得到GPIO5_5的寄存器地址为0x121A_0400;然后GPIO_DIR寄存器里有8位,每一位对应一个GPIO的方向,如果你想把GPIO5_5的方向设为输出,则需要把BIT5置1;

输入命令:himm  0x121A0400  0x20(二进制 0010 0000)

                                         

                                  

3 读取或者写入GPIO值

    GPIO_DATA为GPIO数据寄存器,用来对输入或输出数据进行缓存;当配置GPIO_DIR中对应位为输出时,写入GPIO_DATA 寄存器的值将会输出到相应的管脚(注意需要配置正确的管脚复用);如果配置为输入时,将会读取相应输入管脚的值。

注意:当GPIO_DIR相应的比特配置为输入时,有效读取的结果将返回管脚的值;当配置为输出的时候,有效读取的结果将返回写入的值。GPIO_DATA 寄存器利用PADDR[9:2]实现了读写寄存器比特的屏蔽操作。该寄存器对应256个地址空间。PADDR[9:2]分别对应GPIO_DATA[7:0],当相应的bit 为高时,则可以对相应的位进行读写操作;反之,若对应bit 为低则不能进行操作。

例如:
若地址为0x3FC(0b11_1111_1100),则对GPIO_DATA[7:0]这8bit 操作全部有效。
若地址为0x200(0b10_0000_0000),则仅对GPIO_DATA[7]的操作有效。

同样的,首先需要找地址,对于GPIO5的基地址为:0x121A_0000;GPIO_DATA 的偏移地址为0x3FC,得到GPIO5_5的寄存器地址为0x121A_03FC,如果你想把GPIO5_5写入值为1时,则需要把BIT5置1;

输入命令:himm 0x121A03FC 0x20(二进制 0010 0000)

                               

二、PWM 管脚配置
    上面也说了,对于gpio端口复用功能,可以通过配置复用寄存器muxctrl_reg54,把这个gpio端口配置为PWM_OUT0的功能,配置为PWM管脚功能的歩聚:1、配置gpio复用;2、设置CORE PWM 参数;3、使能CORE 对应的PWM;

1、配置gpio复用

按照上面说明,只需要把muxctrl_reg54复用寄存器配置为0,即为PWM_OUT0功能,

执行命令:himm 0x120F00D8 0x00

2、设置CORE PWM 参数

根据所需的PWM 输出频率和占空比,计算出对应的PWM 计数周期数和高电平计数值。
计数周期数计算公式为:
    pwm_ period = (24000000 / Freq) −1
高电平计数值计算公式为:
    pwm_ duty = (24000000/ Freq)*duty −1

例如,芯片CORE电源调压控制由PWM0信号控制,如果需要PWM0 输出频率为
200KHz, 占空比为75%,配置步骤如下:

Pwm_period=(24000000/200000)-1 = 119
Pwm_duty=(24000000/200000) x 0.75-1=89

设置CORE PWM 参数:设置PERI_PMC0 [15:0]=0x77, PERI_PMC0 [31:16]=0x59;

PCM的基地址为0x120E_0000,PERI_PMC0的偏移地址:0x0000

执行命令: himm 0x120E0000  0x00590077

                           

3、使能CORE 对应的PWM

使能CORE 对应的PWM: 设置PERI_PMC4 [0]=1;

PCM的基地址为0x120E_0000,PERI_PMC4的偏移地址:0x0010

执行命令:himm 0x120E0010 0x01

PERI_PMC4

                                          

                             

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

海思芯片上GPIO和PWM操作 的相关文章

  • 如何处理 C 和 Python 之间的 IPC?

    我有一个具有两个进程的应用程序 一个在 C 中 一个在 Python 中 C 进程是完成所有繁重工作的地方 而 Python 进程则处理用户界面 C 程序每秒向大型缓冲区写入 4 次 Python 进程读取该数据 至此与Python进程的通
  • 如何从子进程(通过 Parallel::ForkManager 分叉)传递变量?

    我的查询 在下面的代码中我试图打印 commandoutput 0 被转移或传递到即将到来的子程序中 我尝试通过转移来传递它 但我失败了 你能帮我正确的方法吗 Code my max forks 4 createThreads my com
  • contextBridge.exposeInMainWorld 和 IPC 在 Electron 应用程序中使用 Typescript:无法读取未定义的属性“发送”

    我定义了 contextBridge https www electronjs org docs all contextbridge https www electronjs org docs all contextbridge 在prel
  • 共享内存和IPC [关闭]

    Closed 这个问题需要多问focused help closed questions 目前不接受答案 我正在阅读有关共享内存的教程 发现以下陈述 如果一个进程希望通知另一个进程新数据已插入共享内存 则必须使用信号 消息队列 管道 套接字
  • boost find in共享内存方法陷入c++多进程项目中

    我正在使用 boost 的 ipc 库来保存复杂的对象 包括图像 在共享内存中 由多个进程使用 我们称这个对象为MyImage 共享内存是一个循环缓冲区 保存了几个MyImage一次对象 在我的代码中 有两个 或更多 进程写入共享内存中的一
  • C# 和 C++ 之间的进程间通信

    我正在为游戏编写一个机器人 它具有 C API 接口 即 当事件发生时 游戏会调用 Cpp dll 中的方法 该 dll 可以回调游戏中的方法来触发操作 我真的不想用 C 编写我的机器人 我是一个相当有经验的 C 程序员 但我根本没有 C
  • 将管道/连接作为上下文参数传递给多处理 Pool.apply_async()

    我想使用管道与池中的流程实例进行通信 但出现错误 让 p 成为 Pool 的一个实例 master pipe worker pipe Pipe self p apply async worker task handler info cont
  • NamedPipe 多个服务器

    对于简单的 IPC 我选择了 NamedPipes 在进程 本地 之间进行通信 由于需求的变化 应该有多个服务器实例 这会导致同一管道名上有多个 侦听器 但似乎有一个问题 这些侦听器中只有一个会收到消息 其他所有实例都不会收到消息 有某种
  • C++ 进程间通信的最佳方式

    我有两个进程 一个进程将查询另一个进程的数据 在有限的时间内 每秒 10000 个 将会有大量的查询 并且每秒将传输数据 gt 100 mb 数据类型将是整数类型 双精度 整数 我的问题是用什么方式连接这个过程 共享内存 消息队列 lpc
  • msgrcv - SA_RESTART 标志不起作用

    我的使用 IPC 队列在线程之间通信的代码有问题 我需要安全地处理 SIGINT 让程序在关闭之前出现 SIGINT 时完成所有活动线程 不过 我在寻找解决方案方面遇到了严重的问题 因为即使使用带有标志 SA RESTART 的 SIGIN
  • 将一个对象传递给默认 AppDomain,以从进程内创建的子 AppDomain 接收回调

    地点 我正在从我的进程创建一个子应用程序域来加载程序集 我能够调用此 AppDomain 我想将一个对象从我的默认进程 AppDomain 传递到这个新创建的 AppDomain 以接收从新 AppDomain 中加载的程序集到我的默认 A
  • 登录方式使用GINA定制

    我知道在 GINA 中找到大师并不容易 但我的问题最接近进程间通信 IPC 我用非托管 c 编写了我的自定义 GINA 我在其中包含了一个方法来检查用户尝试的指纹的有效性为了登录 该函数将调用正在运行的系统Windows服务中用C 编写的一
  • Java/Python 中的快速 IPC/Socket 通信

    我的应用程序中需要两个进程 Java 和 Python 进行通信 我注意到套接字通信占用了 93 的运行时间 为什么通讯这么慢 我应该寻找套接字通信的替代方案还是可以使其更快 更新 我发现了一个简单的修复方法 由于某些未知原因 缓冲输出流似
  • 如何抑制Windows防火墙的Windows安全警报?

    当我从这里找到的 ZeroMQ 指南中用 C 创建 Hello World 示例时 http zguide zeromq org page all Ask and Ye Shall Receive http zguide zeromq or
  • 如何在基于 Linux 的系统上的 C 程序中使用 mqueue?

    如何在基于 Linux 的系统上的 C 程序中使用 mqueue 消息队列 我正在寻找一些好的代码示例 可以展示如何以正确且正确的方式完成此操作 也许是一个操作指南 下面是一个服务器的简单示例 该服务器接收来自客户端的消息 直到收到告诉其停
  • Chrome + 另一个进程:进程间通信比 HTTP/XHR 请求更快?

    我有一个进程 1 对视频流进行实时图像处理 我需要在 Chrome 中的 HTML 页面中渲染该视频 同一台计算机上的进程 2 在canvas or img or videoHTML5 元素 由于我有 1000x1000 像素 x 3 字节
  • Zuul不转发请求到其他微服务

    我正在使用 Spring Boot 微服务 我已经配置了 eureka zuul 代理和另一个微服务 帐户 如果我直接从帐户拨打电话 则工作正常 帐户和 zuul 服务器都显示在 eureka 上 当我尝试使用 zuul 代理进行访问时 它
  • 单台机器最快的 Perl IPC/消息队列是多少?

    我正在开发一个 主要 Perl 项目 并希望使用消息队列来相互隔离进程 我有这样的工作流程 输入 gt 接收器 gt 处理器 gt 输出 我需要每秒处理数百笔交易 所以速度是我最大的动力 对于这种类型的设置来说 最快的消息队列系统是什么 我
  • 在 Windows 容器中使用命名管道(同一主机)

    我想要有 2 个 Windows 容器 在同一主机上运行 使用 Windows 10 客户端计算机和 Windows 的 docker 通过命名管道 不是匿名管道 进行通信 然而 我无法让它发挥作用 我的命名管道服务器类是在 GitHub
  • C# 中的 IPC 机制 - 用法和最佳实践

    不久前我在 Win32 代码中使用了 IPC 临界区 事件和信号量 NET环境下场景如何 是否有任何教程解释所有可用选项以及何时使用以及为什么 微软最近在IPC方面的东西是Windows 通信基础 http en wikipedia org

随机推荐