[PATCH v3] lib: sbi_mpxy: Change MPXY state as per-domain data
Samuel Holland
samuel.holland at sifive.com
Mon Mar 24 20:22:43 PDT 2025
Hi Alvin,
On 2025-03-24 10:13 PM, Alvin Che-Chia Chang(張哲嘉) wrote:
> Hi Samuel,
>
>> -----Original Message-----
>> From: Samuel Holland <samuel.holland at sifive.com>
>> Sent: Tuesday, March 25, 2025 10:26 AM
>> To: Alvin Che-Chia Chang(張哲嘉) <alvinga at andestech.com>;
>> opensbi at lists.infradead.org
>> Cc: anup at brainfault.org; peter.lin at sifive.com; yong.li at intel.com
>> Subject: Re: [PATCH v3] lib: sbi_mpxy: Change MPXY state as per-domain data
>>
>> [EXTERNAL MAIL]
>>
>> Hi Alvin,
>>
>> On 2025-03-24 1:39 AM, Alvin Chang wrote:
>>> OpenSBI supports multiple supervisor domains run on same platform.
>>> When these supervisor domains want to communicate with OpenSBI through
>>> MPXY channels, they will allocate MPXY shared memory from their own
>>> memory regions. Therefore, the MPXY state data structure must be
>>> per-domain and per-hart data structure.
>>>
>>> This commit registers per-domain MPXY state data in sbi_mpxy_init().
>>> The original MPXY state allocated in scratch region is also removed.
>>> We also replace sbi_scratch_thishart_offset_ptr() macro as new
>>> sbi_domain_mpxy_state_thishart_ptr() macro which gets MPXY state from
>>> per-domain data.
>>>
>>> Signed-off-by: Alvin Chang <alvinga at andestech.com>
>>> Reviewed-by: Yu-Chien Peter Lin <peter.lin at sifive.com>
>>> ---
>>> include/sbi/sbi_mpxy.h | 16 +++++++
>>> lib/sbi/sbi_mpxy.c | 100
>> +++++++++++++++++++++++++++++++----------
>>> 2 files changed, 92 insertions(+), 24 deletions(-)
>>>
>>> diff --git a/include/sbi/sbi_mpxy.h b/include/sbi/sbi_mpxy.h index
>>> 9da2791..169f028 100644
>>> --- a/include/sbi/sbi_mpxy.h
>>> +++ b/include/sbi/sbi_mpxy.h
>>> @@ -12,6 +12,7 @@
>>>
>>> #include <sbi/sbi_list.h>
>>>
>>> +struct sbi_domain;
>>> struct sbi_scratch;
>>>
>>> #define SBI_MPXY_MSGPROTO_VERSION(Major, Minor) ((Major << 16) |
>>> Minor) @@ -182,4 +183,19 @@ int sbi_mpxy_send_message(u32
>> channel_id,
>>> u8 msg_id, int sbi_mpxy_get_notification_events(u32 channel_id,
>>> unsigned long *events_len);
>>>
>>> +/**
>>> + * Get per-domain MPXY state pointer for a given domain and HART
>>> +index
>>> + * @param dom pointer to domain
>>> + * @param hartindex the HART index
>>> + *
>>> + * @return per-domain MPXY state pointer for given HART index */
>>> +struct mpxy_state *sbi_domain_get_mpxy_state(struct sbi_domain *dom,
>>> + u32 hartindex);
>>> +
>>> +/** Macro to obtain the current hart's MPXY state pointer in current
>> domain */
>>> +#define sbi_domain_mpxy_state_thishart_ptr() \
>>> + sbi_domain_get_mpxy_state(sbi_domain_thishart_ptr(), \
>>> + current_hartindex())
>>> +
>>> #endif
>>> diff --git a/lib/sbi/sbi_mpxy.c b/lib/sbi/sbi_mpxy.c index
>>> c5d9d1b..4a14c4f 100644
>>> --- a/lib/sbi/sbi_mpxy.c
>>> +++ b/lib/sbi/sbi_mpxy.c
>>> @@ -11,6 +11,7 @@
>>> #include <sbi/sbi_domain.h>
>>> #include <sbi/sbi_error.h>
>>> #include <sbi/sbi_hart.h>
>>> +#include <sbi/sbi_heap.h>
>>> #include <sbi/sbi_platform.h>
>>> #include <sbi/sbi_mpxy.h>
>>> #include <sbi/sbi_scratch.h>
>>> @@ -22,9 +23,6 @@
>>> /** Shared memory size across all harts */ static unsigned long
>>> mpxy_shmem_size = PAGE_SIZE;
>>>
>>> -/** Offset of pointer to MPXY state in scratch space */ -static
>>> unsigned long mpxy_state_offset;
>>> -
>>> /** List of MPXY proxy channels */
>>> static SBI_LIST_HEAD(mpxy_channel_list);
>>>
>>> @@ -89,6 +87,12 @@ struct mpxy_state {
>>> struct mpxy_shmem shmem;
>>> };
>>>
>>> +/** Per domain MPXY state */
>>> +struct domain_mpxy_state_priv {
>>> + /** MPXY state for possible HARTs indexed by hartindex */
>>> + struct mpxy_state
>>> +*hartindex_to_mpxy_state_table[SBI_HARTMASK_MAX_BITS];
>>
>> Please don't hardcode SBI_HARTMASK_MAX_BITS here. You can use
>> sbi_hart_count() to update .data_size before calling
>> sbi_domain_register_data(). Unfortunately, a struct cannot contain a flexible
>> array member by itself, so you'll either need to get rid of the struct or add a
>> dummy member.
>
> It seems we can use zero-length array in struct here.
>
> struct domain_mpxy_state_priv {
> struct mpxy_state *hartindex_to_mpxy_state_table[0];
> };
>
> The value of .data_size would be:
>
> dmspriv.data_size = sizeof(struct mpxy_state *) * sbi_hart_count();
Right, and this is the same expression you would need if there was no struct and
the type of dmsp was `struct mpxy_state **`.
> Do you have better idea?
I don't have a strong opinion. The zero-sized array is a somewhat
deprecated/controversial GCC extension, and using `struct mpxy_state **`
produces similarly complex code, but if you prefer the struct, it should be fine.
Regards,
Samuel
>>
>>> +};
>>> +
>>> /** Disable hart shared memory */
>>> static inline void sbi_mpxy_shmem_disable(struct mpxy_state *ms) {
>>> @@ -171,7 +175,7 @@ bool sbi_mpxy_channel_available(void)
>>>
>>> static void mpxy_std_attrs_init(struct sbi_mpxy_channel *channel) {
>>> - struct mpxy_state *ms =
>> sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> + struct mpxy_state *ms = sbi_domain_mpxy_state_thishart_ptr();
>>> u32 capability = 0;
>>>
>>> /* Reset values */
>>> @@ -240,25 +244,73 @@ int sbi_mpxy_register_channel(struct
>> sbi_mpxy_channel *channel)
>>> return SBI_OK;
>>> }
>>>
>>> -int sbi_mpxy_init(struct sbi_scratch *scratch)
>>> +/** Setup per domain MPXY state data */ static int
>>> +domain_mpxy_state_data_setup(struct sbi_domain *dom,
>>> + struct sbi_domain_data
>> *data,
>>> + void *data_ptr)
>>> {
>>> + struct domain_mpxy_state_priv *dmsp = data_ptr;
>>> struct mpxy_state *ms;
>>> + u32 i;
>>> +
>>> + sbi_hartmask_for_each_hartindex(i, dom->possible_harts) {
>>> + ms = sbi_zalloc(sizeof(*ms));
>>> + if (!ms)
>>> + return SBI_ENOMEM;
>>> +
>>> + /*
>>> + * TODO: Proper support for checking msi support from
>>> + * platform. Currently disable msi and sse and use
>>> + * polling
>>> + */
>>> + ms->msi_avail = false;
>>> + ms->sse_avail = false;
>>>
>>> - mpxy_state_offset = sbi_scratch_alloc_type_offset(struct mpxy_state);
>>> - if (!mpxy_state_offset)
>>> - return SBI_ENOMEM;
>>> + sbi_mpxy_shmem_disable(ms);
>>> +
>>> + dmsp->hartindex_to_mpxy_state_table[i] = ms;
>>> + }
>>>
>>> - /**
>>> - * TODO: Proper support for checking msi support from platform.
>>> - * Currently disable msi and sse and use polling
>>> - */
>>> - ms = sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> - ms->msi_avail = false;
>>> - ms->sse_avail = false;
>>> + return 0;
>>> +}
>>>
>>> - sbi_mpxy_shmem_disable(ms);
>>> +/** Cleanup per domain MPXY state data */ static void
>>> +domain_mpxy_state_data_cleanup(struct sbi_domain *dom,
>>> + struct sbi_domain_data
>> *data,
>>> + void *data_ptr) {
>>> + struct domain_mpxy_state_priv *dmsp = data_ptr;
>>> + u32 i;
>>> +
>>> + sbi_hartmask_for_each_hartindex(i, dom->possible_harts)
>>> + sbi_free(dmsp->hartindex_to_mpxy_state_table[i]);
>>> +}
>>> +
>>> +static struct sbi_domain_data dmspriv = {
>>> + .data_size = sizeof(struct domain_mpxy_state_priv),
>>> + .data_setup = domain_mpxy_state_data_setup,
>>> + .data_cleanup = domain_mpxy_state_data_cleanup, };
>>> +
>>> +struct mpxy_state *sbi_domain_get_mpxy_state(struct sbi_domain *dom,
>>> + u32 hartindex) {
>>> + struct domain_mpxy_state_priv *dmsp =
>>> + sbi_domain_data_ptr(dom, &dmspriv);
>>> +
>>> + return (dmsp && hartindex < SBI_HARTMASK_MAX_BITS) ?
>>
>> sbi_hartindex_valid() would be appropriate here.
>
> Will use sbi_hartindex_valid() here, thanks!
>
>
> Regards,
> Alvin
>
>>
>> Regards,
>> Samuel
>>
>>> + dmsp->hartindex_to_mpxy_state_table[hartindex] : NULL; }
>>> +
>>> +int sbi_mpxy_init(struct sbi_scratch *scratch) {
>>> + int ret;
>>> +
>>> + ret = sbi_platform_mpxy_init(sbi_platform_ptr(scratch));
>>> + if (ret)
>>> + return ret;
>>>
>>> - return sbi_platform_mpxy_init(sbi_platform_ptr(scratch));
>>> + return sbi_domain_register_data(&dmspriv);
>>> }
>>>
>>> unsigned long sbi_mpxy_get_shmem_size(void) @@ -270,7 +322,7 @@ int
>>> sbi_mpxy_set_shmem(unsigned long shmem_phys_lo,
>>> unsigned long shmem_phys_hi,
>>> unsigned long flags) {
>>> - struct mpxy_state *ms =
>> sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> + struct mpxy_state *ms = sbi_domain_mpxy_state_thishart_ptr();
>>> unsigned long *ret_buf;
>>>
>>> /** Disable shared memory if both hi and lo have all bit 1s */
>>> @@ -312,7 +364,7 @@ int sbi_mpxy_set_shmem(unsigned long
>>> shmem_phys_lo,
>>>
>>> int sbi_mpxy_get_channel_ids(u32 start_index) {
>>> - struct mpxy_state *ms =
>> sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> + struct mpxy_state *ms = sbi_domain_mpxy_state_thishart_ptr();
>>> u32 remaining, returned, max_channelids;
>>> u32 node_index = 0, node_ret = 0;
>>> struct sbi_mpxy_channel *channel; @@ -363,7 +415,7 @@ int
>>> sbi_mpxy_get_channel_ids(u32 start_index)
>>>
>>> int sbi_mpxy_read_attrs(u32 channel_id, u32 base_attr_id, u32
>>> attr_count) {
>>> - struct mpxy_state *ms =
>> sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> + struct mpxy_state *ms = sbi_domain_mpxy_state_thishart_ptr();
>>> int ret = SBI_SUCCESS;
>>> u32 *attr_ptr, end_id;
>>> void *shmem_base;
>>> @@ -479,7 +531,7 @@ static int mpxy_check_write_std_attr(struct
>>> sbi_mpxy_channel *channel, static void mpxy_write_std_attr(struct
>> sbi_mpxy_channel *channel, u32 attr_id,
>>> u32 attr_val) {
>>> - struct mpxy_state *ms =
>> sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> + struct mpxy_state *ms = sbi_domain_mpxy_state_thishart_ptr();
>>> struct sbi_mpxy_channel_attrs *attrs = &channel->attrs;
>>>
>>> switch(attr_id) {
>>> @@ -513,7 +565,7 @@ static void mpxy_write_std_attr(struct
>>> sbi_mpxy_channel *channel, u32 attr_id,
>>>
>>> int sbi_mpxy_write_attrs(u32 channel_id, u32 base_attr_id, u32
>>> attr_count) {
>>> - struct mpxy_state *ms =
>> sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> + struct mpxy_state *ms = sbi_domain_mpxy_state_thishart_ptr();
>>> u32 *mem_ptr, attr_id, end_id, attr_val;
>>> struct sbi_mpxy_channel *channel;
>>> int ret, mem_idx;
>>> @@ -603,7 +655,7 @@ int sbi_mpxy_send_message(u32 channel_id, u8
>> msg_id,
>>> unsigned long msg_data_len,
>>> unsigned long *resp_data_len) {
>>> - struct mpxy_state *ms =
>> sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> + struct mpxy_state *ms = sbi_domain_mpxy_state_thishart_ptr();
>>> struct sbi_mpxy_channel *channel;
>>> void *shmem_base, *resp_buf;
>>> u32 resp_bufsize;
>>> @@ -661,7 +713,7 @@ int sbi_mpxy_send_message(u32 channel_id, u8
>>> msg_id,
>>>
>>> int sbi_mpxy_get_notification_events(u32 channel_id, unsigned long
>>> *events_len) {
>>> - struct mpxy_state *ms =
>> sbi_scratch_thishart_offset_ptr(mpxy_state_offset);
>>> + struct mpxy_state *ms = sbi_domain_mpxy_state_thishart_ptr();
>>> struct sbi_mpxy_channel *channel;
>>> void *eventsbuf, *shmem_base;
>>> int ret;
More information about the opensbi
mailing list