全志R128 SDK HAL 模块开发指南之 MSGBOX

2024-05-15 11:04

本文主要是介绍全志R128 SDK HAL 模块开发指南之 MSGBOX,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

MSGBOX

msgbox 用来实现多 CPU 之间通讯,在一些 IC 内部可能同时存在多种核心用来实现多种不同功能,这些不同核心运行不同架构、不同系统,需要通过 MSGBOX 用来实现这些不同系统间通讯。

模块介绍

  • msgbox 为一个双端 fifo 结构,cpu0 从一端写,cpu1 从一端读。
  • rpmsg 为 linux 用来实现通讯的一种框架
  • msgbox 为片上处理器之间提供了中断通讯机制

对于 R128 平台,CPU Remote ID如下

CPURemote ID
ARM M33 Star0
RISC-V C9061
HIFI5 DSP2

模块配置

配置路径如下:

Kernel Setup --->Drivers Setup --->SoC HAL Drivers --->msgbox devices --->[*] enable msgbox driver

源码结构

msgbox/├── msgbox_amp            // msgbox AMP 实现│   ├── Makefile│   └── msgbox_amp.c        ├── platform              // 平台寄存器定义│   ├── msgbox-sun20iw2.h└── platform-msgbox.h     // 公共头文件

模块接口说明

头文件

#include <hal_msgbox.h>

初始化接口

函数原型:

int32_t hal_msgbox_init(void);

参数:

返回值:

  • 0:成功
  • 负数:失败

通道申请接口

函数原型:

uint32_t hal_msgbox_alloc_channel(struct msg_endpoint *edp, int32_t remote, int32_t read, int32_t write);

参数:

  • edp:msgbox的端点
  • remote:远端核心id
  • read:读通道
  • write:写通道

返回值:

  • 0:成功
  • 负数:失败

数据发送接口

函数原型:

uint32_t hal_msgbox_channel_send(struct msg_endpoint *edp, uint8_t *bf, uint32_t len);

参数:

  • edp:msgbox的端点
  • bf:数据buffer
  • len:buffer长度

返回值:

  • 0:成功
  • 负数:失败

通道释放接口

函数原型:

void hal_msgbox_free_channel(struct msg_endpoint *edp);

参数:

  • edp:msgbox的端点

返回值:

  • 0:成功
  • 负数:失败

MSGBOX 申请流程

  1. 使用hal_msgbox_alloc_channel接口申请 msgbox 通道
  2. 填充msg_endpoint接收回调,这个会在 msgbox 的中断函数里调用
  3. 通过hal_msgbox_channel_send进行数据发送
  4. 接收通过中断的方式进行接收,会调用msg_endpoint的回调,无需主动调用

MSGBOX 接收流程

  1. 在接收函数里会首先遍历所有的msg_endpoint,判断当前终端是否有中断发送
  2. irq_msgbox_channel_handler里会读取当前msg_endpoint的寄存器,来判断是否有中断,如果有,则读取数据
  3. 退出中断

MSGBOX 发送流程

  1. 调用hal_msgbox_channel_send接口进行数据发送
  2. msgbox_channel_send_data会判断是远端处理器是哪个,并且计算 local->remote 的系数 N 是多少,这个系数回存放在 to_coef_n 的表格里
  3. 计算完成后往远端的 msgbox 的 fifo 中写数据
  4. 发送完成

模块使用范例

#include <FreeRTOS.h>
#include <queue.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <unistd.h>
#include <string.h>
#include <awlog.h>
#include <hal_msgbox.h>#include <console.h>#define RECEIVE_QUEUE_LENGTH 16
#define RECEIVE_QUEUE_WAIT_MS 100struct msgbox_demo {int remote_id;int read_ch;int write_ch;QueueHandle_t recv_queue;
};static void print_help_msg(void)
{printf("\n");printf("USAGE:\n");printf("  hal_msgbox <REQUIRED_ARGUMENTS> [OPTIONS]\n");printf("\n");printf("REQUIRED_ARGUMENTS:\n");printf("  -E REMOTE_ID: specify remote id\n");printf("  -R READ_CH  : specify read channel\n");printf("  -W WRITE_CH : specify write channel\n");printf("OPTIONS:\n");printf("  -s MESSAGE  : send MESSAGE\n");printf("  -r          : receive messages\n");printf("  -t TIMEOUT  : exit in TIMEOUT seconds when receive\n");printf("e.g. (communicate with remote 0, use read channel 3 and write channel 3):\n");printf("  hal_msgbox -E 0 -R 3 -W 3 -s \"hello\" : send string \"hello\"\n");printf("  hal_msgbox -E 0 -R 3 -W 3 -r           : receive messages (default in 10 seconds)\n");printf("  hal_msgbox -E 0 -R 3 -W 3 -r -t 20     : receive messages in 20 seconds\n");printf("\n");
}static int recv_callback(unsigned long data, void *private_data)
{BaseType_t taskwoken = pdFALSE;printf("Receive callback (data: 0x%lx)\n", data);struct msgbox_demo *demo = private_data;BaseType_t ret = xQueueSendFromISR(demo->recv_queue, &data, &taskwoken);if (ret == errQUEUE_FULL) {printf("recv_queue is full\n");return -1;}if (ret == pdPASS) {portYIELD_FROM_ISR(taskwoken);}return 0;
}static int cmd_hal_msgbox(int argc, char *argv[])
{int ret = 0;int c;struct msgbox_demo demo= {.remote_id = -1,.read_ch = -1,.write_ch = -1,.recv_queue = NULL,};struct msg_endpoint ept;TickType_t start_ticks, current_ticks;int do_send = 0;const char *data_send= NULL;int do_recv = 0;int timeout_sec = 10;uint32_t data_recv;if (argc <= 1) {print_help_msg();ret = -1;goto out;}while ((c = getopt(argc, argv, "hs:rt:E:W:R:")) != -1) {switch (c) {case 'h' :print_help_msg();ret = 0;goto out;case 'E':demo.remote_id = atoi(optarg);break;case 'R':demo.read_ch = atoi(optarg);break;case 'W':demo.write_ch = atoi(optarg);break;case 's':do_send = 1;data_send = optarg;break;case 'r':do_recv = 1;break;case 't':timeout_sec = atoi(optarg);break;default:print_help_msg();ret = -1;goto out;}}if (demo.remote_id < 0 || demo.read_ch < 0 || demo.write_ch < 0) {printf("Error. Please specify remote id, read channel and write channel\n");print_help_msg();ret = -1;goto out;}printf("remote id: %d, write channel: %d, read channel: %d\n",demo.remote_id, demo.write_ch, demo.read_ch);if (do_recv) {demo.recv_queue = xQueueCreate(RECEIVE_QUEUE_LENGTH, sizeof(uint32_t));if (!demo.recv_queue) {printf("Failed to create receive queue\n");ret = -1;goto out;}ept.rec = (void *)recv_callback;ept.private = &demo;}ret = hal_msgbox_alloc_channel(&ept, demo.remote_id, demo.read_ch, demo.write_ch);if (ret != 0) {printf("Failed to allocate msgbox channel\n");goto delete_recv_queue;}if (do_send) {ret = hal_msgbox_channel_send(&ept, (unsigned char *)data_send, strlen(data_send));if (ret != 0) {printf("Failed to send message\n");goto free_channel;}}if (do_recv) {printf("hal_msgbox will exit in %d seconds\n", timeout_sec);start_ticks = xTaskGetTickCount();printf("start_ticks: %u\n", start_ticks);while (1) {if (pdTRUE == xQueueReceive(demo.recv_queue, &data_recv,RECEIVE_QUEUE_WAIT_MS / portTICK_PERIOD_MS)) {printf("Received from queue: 0x%x\n", data_recv);}current_ticks = xTaskGetTickCount();if ((current_ticks - start_ticks) * portTICK_PERIOD_MS>= timeout_sec * 1000) {printf("current_ticks: %u\n", current_ticks);break;}}}printf("hal_msgbox exited\n");ret = 0;free_channel:hal_msgbox_free_channel(&ept);
delete_recv_queue:if (do_recv) {vQueueDelete(demo.recv_queue);}
out:return ret;
}FINSH_FUNCTION_EXPORT_CMD(cmd_hal_msgbox, hal_msgbox, hal msgbox);

这篇关于全志R128 SDK HAL 模块开发指南之 MSGBOX的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/991657

相关文章

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

python: 多模块(.py)中全局变量的导入

文章目录 global关键字可变类型和不可变类型数据的内存地址单模块(单个py文件)的全局变量示例总结 多模块(多个py文件)的全局变量from x import x导入全局变量示例 import x导入全局变量示例 总结 global关键字 global 的作用范围是模块(.py)级别: 当你在一个模块(文件)中使用 global 声明变量时,这个变量只在该模块的全局命名空

Hadoop企业开发案例调优场景

需求 (1)需求:从1G数据中,统计每个单词出现次数。服务器3台,每台配置4G内存,4核CPU,4线程。 (2)需求分析: 1G / 128m = 8个MapTask;1个ReduceTask;1个mrAppMaster 平均每个节点运行10个 / 3台 ≈ 3个任务(4    3    3) HDFS参数调优 (1)修改:hadoop-env.sh export HDFS_NAMENOD

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

Retrieval-based-Voice-Conversion-WebUI模型构建指南

一、模型介绍 Retrieval-based-Voice-Conversion-WebUI(简称 RVC)模型是一个基于 VITS(Variational Inference with adversarial learning for end-to-end Text-to-Speech)的简单易用的语音转换框架。 具有以下特点 简单易用:RVC 模型通过简单易用的网页界面,使得用户无需深入了

Java 创建图形用户界面(GUI)入门指南(Swing库 JFrame 类)概述

概述 基本概念 Java Swing 的架构 Java Swing 是一个为 Java 设计的 GUI 工具包,是 JAVA 基础类的一部分,基于 Java AWT 构建,提供了一系列轻量级、可定制的图形用户界面(GUI)组件。 与 AWT 相比,Swing 提供了许多比 AWT 更好的屏幕显示元素,更加灵活和可定制,具有更好的跨平台性能。 组件和容器 Java Swing 提供了许多

Linux_kernel驱动开发11

一、改回nfs方式挂载根文件系统         在产品将要上线之前,需要制作不同类型格式的根文件系统         在产品研发阶段,我们还是需要使用nfs的方式挂载根文件系统         优点:可以直接在上位机中修改文件系统内容,延长EMMC的寿命         【1】重启上位机nfs服务         sudo service nfs-kernel-server resta