Recent half of year, I made development on Qualcomm platform FSM990x, which is ARM A7 based, same core with Mobile phone a few years ago, but equipped with more powerful DSP and FPGA capability for special purpose.
In the Qualcomm SDK, there are some concepts which need to be learned with some effort. I had like share my thoughts on SMEM, SMD,SMSM and QMI in following series of articles.
In the Kernel tree, there is a brief introduction on SMEM,SMD,SMSM in the soc/qcom/qcom,smem.txt, But it has little hard to understand what is the purpose and usage on SMEM,SMD,SMSM in Qualcomm platform, I’d like share my thought on this topic.
@SMEM
Qualcomm Shared Memory Manager binding
This binding describes the Qualcomm Shared Memory Manager, used to share data
between various subsystems and OSes in Qualcomm platforms.
@SMD
Qualcomm Shared Memory Driver (SMD) binding
This binding describes the Qualcomm Shared Memory Driver, a fifo based
communication channel for sending data between the various subsystems in
Qualcomm platforms.
Qualcomm Soc has some heterogeneous process units inside, like DSP, Modem, etc. For better message transfer, Qualcomm use shared_memory and interrupt to interchange the control message (not the user plane message, as SMEM is not purposed on that, Qualcomm provide HW FIFO or DMA on that purpose which provide more throughput than SMEM based method like QMI, etc. It also not suitable to use SMEM to interchange the message requiring high throughput? – I think it still practical to use shared_memory to interchange high throughput data with PUSH-PULL model instead of SMEM message notification scheme, anyway HW support FIFO it has some other advantages over shared_memory).
The Interface exposed by SMEM:
enum id{
/* fixed items */
SMEM_PROC_COMM = 0,
SMEM_HEAP_INFO,
SMEM_ALLOCATION_TABLE,
SMEM_VERSION_INFO,
SMEM_HW_RESET_DETECT,
SMEM_AARM_WARM_BOOT,
....
/* dynamic items */
...
SMEM_AARM_PARTITION_TABLE,
SMEM_AARM_BAD_BLOCK_TABLE,
SMEM_ERR_CRASH_LOG_ADSP,
SMEM_WM_UUID,
SMEM_CHANNEL_ALLOC_TBL,
SMEM_SMD_BASE_ID,
...
}
void *smem_alloc(enum id, unsigned size_in, unsigned to_proc,
unsigned flags);
void *smem_find(enum id, unsigned size_in, unsigned to_proc,
unsigned flags);
@id is ID of SEME item;
@size_in size of the SMEM item;
@to_proc SMEM host that shares the item with apps, in the host, SMEM make partitions for each SUBSYSTEM such like, DSP, Modem etc, in each partition they have same allocation patterns, but in some scenarios, FLAG with SMEM_ANY_HOST_FLAG will indicatethe smem_alloc to use the shared space in top of the space for allocation. SMEM_ANY_HOST_FLAG means this space was depended with any subsystems and can be used by any partition.
@flags Item attribute flags
/*
* Flag options for the XXX_to_proc() API
*
* SMEM_ITEM_CACHED_FLAG - Indicates this operation should use cachable smem
*
* SMEM_ANY_HOST_FLAG - Indicates this operation should not apply to smem items
* which are limited to a specific host pairing. Will
* cause this operation to ignore the to_proc parameter.
*/
\#define SMEM_ITEM_CACHED_FLAG 1
\#define SMEM_ANY_HOST_FLAG 2
SMEM maint a cached and uncached space in the secute space, the cacheable space was aligned with the cache size in the header declared, my guess is because some subsystems were required with alignment of the memory. As far as SMEM_ANY_HOST_FLAG, I had already explained in the last section.
{% asset_img “smem.png” “c++code” %}
smem platform driver use Device Tree to parse the memory layout. f.g.
qcom,smem@13600000 {
compatible = "qcom,smem";
reg = <0x13600000 0x200000>,
<0xf9011000 0x1000>;
reg-names = "smem", "irq-reg-base";
qcom,smd-hex0 {
compatible = "qcom,smd";
qcom,smd-edge = <0>;
qcom,smd-irq-offset = <0x8>;
qcom,smd-irq-bitmask = <0x100>;
interrupts = <0 72 1>;
label = "hex0";
qcom,not-loadable;
};
qcom,smd-hex1 {
compatible = "qcom,smd";
qcom,smd-edge = <1>;
qcom,smd-irq-offset = <0x8>;
qcom,smd-irq-bitmask = <0x1000>;
interrupts = <0 68 1>;
label = "hex1";
qcom,not-loadable;
};
qcom,smd-tenx {
compatible = "qcom,smd";
qcom,smd-edge = <10>;
qcom,smd-irq-offset = <0x8>;
qcom,smd-irq-bitmask = <0x10>;
interrupts = <0 26 1>;
label = "tenx";
qcom,not-loadable;
};
};
From the node above, dt declare a qcom,smem node with memory phy_addr-size {0x13600000-0x200000} and IRQ reg phy_addr-size {0xf9011000-0x1000}. The driver use ioremap_nocache(…) to map to virtual address.
In some SoC variants, Qualcomm declare aux-%1 in dts file, smem reserve the spaces for them when co-processor happened RESTART it can dump the ELF file image aka. ramdump. In such cases, SMEM will create the stucture to describe them as well.
SMEM part 2 different zones for whole available ram space, SMEM define a enum to describe that fixed items should be allocated in security zone, and dynamic item should be allocated in non-security zone.
I had like make a compare with other SoC vendor how they handle such IPC between heterogeneous system, as far as I know, like TI, Xilinx currently they would like use RPMSG here based on virtio as backend. RPMSG is also a shared memory (virtqueue) based tech.
Qualcomm use SMEM as backend to extend some basic services like SMD, SMEM etc.
I will talk about that in the next ariticle.