* [PATCH net-next v22 0/2] MCTP Over PCC Transport @ 2025-07-10 19:12 admiyo 2025-07-10 19:12 ` [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer admiyo 2025-07-10 19:12 ` [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport admiyo 0 siblings, 2 replies; 10+ messages in thread From: admiyo @ 2025-07-10 19:12 UTC (permalink / raw) Cc: netdev, linux-kernel, Jeremy Kerr, Matt Johnston, David S . Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni, Sudeep Holla, Jonathan Cameron, Huisong Li From: Linux Bot <linuxbot@amperecomputing.com> This series adds support for the Management Control Transport Protocol (MCTP) over the Platform Communication Channel (PCC) mechanism. DMTF DSP:0292 https://www.dmtf.org/sites/default/files/standards/documents/DSP0292_1.0.0WIP50.pdf MCTP defines a communication model intended to facilitate communication between Management controllers and other management controllers, and between Management controllers and management devices PCC is a mechanism for communication between components within the Platform. It is a composed of shared memory regions, interrupt registers, and status registers. The MCTP over PCC driver makes use of two PCC channels. For sending messages, it uses a Type 3 channel, and for receiving messages it uses the paired Type 4 channel. The device and corresponding channels are specified via ACPI. The first patch in the series implements a mechanism to allow the driver to indicate whether an ACK should be sent back to the caller after processing the interrupt. This is an optional feature in the PCC code, but has been made explicitly required in another driver. The implementation here maintains the backwards compatibility of that driver. MCTP is a general purpose protocol so it would be impossible to enumerate all the use cases, but some of the ones that are most topical are attestation and RAS support. There are a handful of protocols built on top of MCTP, to include PLDM and SPDM, both specified by the DMTF. https://www.dmtf.org/sites/default/files/standards/documents/DSP0240_1.0.0.pdf https://www.dmtf.org/sites/default/files/standards/documents/DSP0274_1.3.0.pd SPDM entails various usages, including device identity collection, device authentication, measurement collection, and device secure session establishment. PLDM is more likely to be used for hardware support: temperature, voltage, or fan sensor control. At least two companies have devices that can make use of the mechanism. One is Ampere Computing, my employer. The mechanism it uses is called Platform Communication Channels is part of the ACPI spec: https://uefi.org/htmlspecs/ACPI_Spec_6_4_html/14_Platform_Communications_Channel/Platform_Comm_Channel.html Since it is a socket interface, the system administrator also has the ability to ignore an MCTP link that they do not want to enable. This link would be visible to the end user, but would not be usable. If MCTP support is disabled in the Kernel, this driver would also be disabled. PCC is based on a shared buffer and a set of I/O mapped memory locations that the Spec calls registers. This mechanism exists regardless of the existence of the driver. Thus, if the user has the ability to map these physical location to virtual locations, they have the ability to drive the hardware. Thus, there is a security aspect to this mechanism that extends beyond the responsibilities of the operating system. If the hardware does not expose the PCC in the ACPI table, this device will never be enabled. Thus it is only an issue on hard that does support PCC. In that case, it is up to the remote controller to sanitize communication; MCTP will be exposed as a socket interface, and userland can send any crafted packet it wants. It would thus also be incumbent on the hardware manufacturer to allow the end user to disable MCTP over PCC communication if they did not want to expose it. Previous implementations of the pcc version of the mailbox protocol assumed the driver was directly managing the shared memory region. This lead to duplicated code and missed stpes of the PCC protocol. The first patch in this series makes it possible for mailbox/pcc to manage the writing of the buffer prior to sending messages. It also fixes the notification of message transmission completion. Previous Version: https://lore.kernel.org/lkml/20250429222759.138627-1-admiyo@os.amperecomputing.com/ Changes in V22: - Direct management of the shared buffer in the mailbox layer. - Proper checking of command complete flag prior to writing to the buffer. Changes in V21: - Use existing constants PCC_SIGNATURE and PCC_CMD_COMPLETION_NOTIFY - Check return code on call to send_data and drop packet if failed - use sizeof(*mctp_pcc_header) etc, instead of structs for resizing buffers - simplify check for ares->type != PCC_DWORD_TYPE - simply return result devm_add_action_or_reset - reduce initializer for mctp_pcc_lookup_context context = {}; - move initialization of mbox dev into mctp_pcc_initialize_mailbox - minor spacing changes Changes in V20: - corrected typo in RFC version - removed spurious space - tx spin lock only controls access to shared memory buffer - tx spin lock not eheld on error condition - tx returns OK if skb can't be expanded Changes in V19: - Rebased on changes to PCC mailbox handling - checks for cloned SKB prior to transmission - converted doulbe slash comments to C comments Changes in V18: - Added Acked-By - Fix minor spacing issue Changes in V17: - No new changes. Rebased on net-next post 6.13 release. Changes in V16: - do not duplicate cleanup after devm_add_action_or_reset calls Changes in V15: - corrected indentation formatting error - Corrected TABS issue in MAINTAINER entry Changes in V14: - Do not attempt to unregister a netdev that is never registered - Added MAINTAINER entry Changes in V13: - Explicitly Convert PCC header from little endian to machine native Changes in V12: - Explicitly use little endian conversion for PCC header signature - Builds clean with make C=1 Changes in V11: - Explicitly use little endian types for PCC header Changes in V11: - Switch Big Endian data types to machine local for PCC header - use mctp specific function for registering netdev Changes in V10: - sync with net-next branch - use dstats helper functions - remove duplicate drop stat - remove more double spaces Changes in V9: - Prerequisite patch for PCC mailbox has been merged - Stats collection now use helper functions - many double spaces reduced to single Changes in V8: - change 0 to NULL for pointer check of shmem - add semi for static version of pcc_mbox_ioremap - convert pcc_mbox_ioremap function to static inline when client code is not being built - remove shmem comment from struct pcc_chan_info descriptor - copy rx_dropped in mctp_pcc_net_stats - removed trailing newline on error message - removed double space in dev_dbg string - use big endian for header members - Fix use full spec ID in description - Fix typo in file description - Form the complete outbound message in the sk_buff Changes in V7: - Removed the Hardware address as specification is not published. - Map the shared buffer in the mailbox and share the mapped region with the driver - Use the sk_buff memory to prepare the message before copying to shared region Changes in V6: - Removed patch for ACPICA code that has merged - Includes the hardware address in the network device - Converted all device resources to devm resources - Removed mctp_pcc_driver_remove function - uses acpi_driver_module for initialization - created helper structure for in and out mailboxes - Consolidated code for initializing mailboxes in the add_device function - Added specification references - Removed duplicate constant PCC_ACK_FLAG_MASK - Use the MCTP_SIGNATURE_LENGTH define - made naming of header structs consistent - use sizeof local variables for offset calculations - prefix structure name to avoid potential clash - removed unnecessary null initialization from acpi_device_id Changes in V5 - Removed Owner field from ACPI module declaration - removed unused next field from struct mctp_pcc_ndev - Corrected logic reading RX ACK flag. - Added comment for struct pcc_chan_info field shmem_base_addr - check against current mtu instead of max mtu for packet length\ - removed unnecessary lookups of pnd->mdev.dev Changes in V4 - Read flags out of shared buffer to trigger ACK for Type 4 RX - Remove list of netdevs and cleanup from devices only - tag PCCT protocol headers as little endian - Remove unused constants Changes in V3 - removed unused header - removed spurious space - removed spurious semis after functiomns - removed null assignment for init - remove redundant set of device on skb - tabify constant declarations - added rtnl_link_stats64 function - set MTU to minimum to start - clean up logic on driver removal - remove cast on void * assignment - call cleanup function directly - check received length before allocating skb - introduce symbolic constatn for ACK FLAG MASK - symbolic constant for PCC header flag. - Add namespace ID to PCC magic - replaced readls with copy from io of PCC header - replaced custom modules init and cleanup with ACPI version Changes in V2 - All Variable Declarations are in reverse Xmass Tree Format - All Checkpatch Warnings Are Fixed - Removed Dead code - Added packet tx/rx stats - Removed network physical address. This is still in disucssion in the spec, and will be added once there is consensus. The protocol can be used with out it. This also lead to the removal of the Big Endian conversions. - Avoided using non volatile pointers in copy to and from io space - Reorderd the patches to put the ACK check for the PCC Mailbox as a pre-requisite. The corresponding change for the MCTP driver has been inlined in the main patch. - Replaced magic numbers with constants, fixed typos, and other minor changes from code review. Adam Young (2): mailbox/pcc: support mailbox management of the shared buffer mctp pcc: Implement MCTP over PCC Transport MAINTAINERS | 5 + drivers/mailbox/pcc.c | 91 +++++++++- drivers/net/mctp/Kconfig | 13 ++ drivers/net/mctp/Makefile | 1 + drivers/net/mctp/mctp-pcc.c | 346 ++++++++++++++++++++++++++++++++++++ include/acpi/pcc.h | 19 ++ 6 files changed, 472 insertions(+), 3 deletions(-) create mode 100644 drivers/net/mctp/mctp-pcc.c -- 2.43.0 ^ permalink raw reply [flat|nested] 10+ messages in thread
* [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer 2025-07-10 19:12 [PATCH net-next v22 0/2] MCTP Over PCC Transport admiyo @ 2025-07-10 19:12 ` admiyo 2025-07-11 6:51 ` kernel test robot ` (2 more replies) 2025-07-10 19:12 ` [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport admiyo 1 sibling, 3 replies; 10+ messages in thread From: admiyo @ 2025-07-10 19:12 UTC (permalink / raw) To: Sudeep Holla, Jassi Brar, Robert Moore, Rafael J. Wysocki, Len Brown Cc: netdev, linux-kernel, Jeremy Kerr, Matt Johnston, David S . Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni, Jonathan Cameron, Huisong Li From: Adam Young <admiyo@os.amperecomputing.com> Define a new, optional, callback that allows the driver to specify how the return data buffer is allocated. If that callback is set, mailbox/pcc.c is now responsible for reading from and writing to the PCC shared buffer. This also allows for proper checks of the Commnand complete flag between the PCC sender and receiver. For Type 4 channels, initialize the command complete flag prior to accepting messages. Since the mailbox does not know what memory allocation scheme to use for response messages, the client now has an optional callback that allows it to allocate the buffer for a response message. When an outbound message is written to the buffer, the mailbox checks for the flag indicating the client wants an tx complete notification via IRQ. Upon reciept of the interrupt It will pair it with the outgoing message. The expected use is to free the kernel memory buffer for the previous outgoing message. Signed-off-by: Adam Young <admiyo@os.amperecomputing.com> --- drivers/mailbox/pcc.c | 91 +++++++++++++++++++++++++++++++++++++++++-- include/acpi/pcc.h | 19 +++++++++ 2 files changed, 107 insertions(+), 3 deletions(-) diff --git a/drivers/mailbox/pcc.c b/drivers/mailbox/pcc.c index f6714c233f5a..2932c26aaf62 100644 --- a/drivers/mailbox/pcc.c +++ b/drivers/mailbox/pcc.c @@ -306,6 +306,22 @@ static void pcc_chan_acknowledge(struct pcc_chan_info *pchan) pcc_chan_reg_read_modify_write(&pchan->db); } +static void *write_response(struct pcc_chan_info *pchan) +{ + struct pcc_header pcc_header; + void *buffer; + int data_len; + + memcpy_fromio(&pcc_header, pchan->chan.shmem, + sizeof(pcc_header)); + data_len = pcc_header.length - sizeof(u32) + sizeof(struct pcc_header); + + buffer = pchan->chan.rx_alloc(pchan->chan.mchan->cl, data_len); + if (buffer != NULL) + memcpy_fromio(buffer, pchan->chan.shmem, data_len); + return buffer; +} + /** * pcc_mbox_irq - PCC mailbox interrupt handler * @irq: interrupt number @@ -317,6 +333,7 @@ static irqreturn_t pcc_mbox_irq(int irq, void *p) { struct pcc_chan_info *pchan; struct mbox_chan *chan = p; + void *handle = NULL; pchan = chan->con_priv; @@ -340,7 +357,14 @@ static irqreturn_t pcc_mbox_irq(int irq, void *p) * required to avoid any possible race in updatation of this flag. */ pchan->chan_in_use = false; - mbox_chan_received_data(chan, NULL); + + if (pchan->chan.rx_alloc) + handle = write_response(pchan); + + if (pchan->chan.irq_ack) + mbox_chan_txdone(chan, 0); + + mbox_chan_received_data(chan, handle); pcc_chan_acknowledge(pchan); @@ -384,9 +408,22 @@ pcc_mbox_request_channel(struct mbox_client *cl, int subspace_id) pcc_mchan = &pchan->chan; pcc_mchan->shmem = acpi_os_ioremap(pcc_mchan->shmem_base_addr, pcc_mchan->shmem_size); - if (pcc_mchan->shmem) - return pcc_mchan; + if (!pcc_mchan->shmem) + goto err; + + /* This indicates that the channel is ready to accept messages. + * This needs to happen after the channel has registered + * its callback. There is no access point to do that in + * the mailbox API. That implies that the mailbox client must + * have set the allocate callback function prior to + * sending any messages. + */ + if (pchan->type == ACPI_PCCT_TYPE_EXT_PCC_SLAVE_SUBSPACE) + pcc_chan_reg_read_modify_write(&pchan->cmd_update); + + return pcc_mchan; +err: mbox_free_channel(chan); return ERR_PTR(-ENXIO); } @@ -417,6 +454,27 @@ void pcc_mbox_free_channel(struct pcc_mbox_chan *pchan) } EXPORT_SYMBOL_GPL(pcc_mbox_free_channel); +static int pcc_write_to_buffer(struct mbox_chan *chan, void *data) +{ + struct pcc_chan_info *pchan = chan->con_priv; + struct pcc_mbox_chan *pcc_mbox_chan = &pchan->chan; + struct pcc_header *pcc_header = data; + /* The PCC header length includes the command field + * but not the other values from the header. + */ + int len = pcc_header->length - sizeof(u32) + sizeof(struct pcc_header); + u64 val; + + pcc_chan_reg_read(&pchan->cmd_complete, &val); + if (!val) { + pr_info("%s pchan->cmd_complete not set", __func__); + return -1; + } + memcpy_toio(pcc_mbox_chan->shmem, data, len); + return 0; +} + + /** * pcc_send_data - Called from Mailbox Controller code. Used * here only to ring the channel doorbell. The PCC client @@ -433,18 +491,44 @@ static int pcc_send_data(struct mbox_chan *chan, void *data) { int ret; struct pcc_chan_info *pchan = chan->con_priv; + struct acpi_pcct_ext_pcc_shared_memory __iomem *pcc_hdr; + + if (pchan->chan.rx_alloc) + ret = pcc_write_to_buffer(chan, data); + if (ret) + return ret; ret = pcc_chan_reg_read_modify_write(&pchan->cmd_update); if (ret) return ret; + pcc_hdr = pchan->chan.shmem; + if (ioread32(&pcc_hdr->flags) & PCC_CMD_COMPLETION_NOTIFY) + pchan->chan.irq_ack = true; + ret = pcc_chan_reg_read_modify_write(&pchan->db); + if (!ret && pchan->plat_irq > 0) pchan->chan_in_use = true; return ret; } + +static bool pcc_last_tx_done(struct mbox_chan *chan) +{ + struct pcc_chan_info *pchan = chan->con_priv; + u64 val; + + pcc_chan_reg_read(&pchan->cmd_complete, &val); + if (!val) + return false; + else + return true; +} + + + /** * pcc_startup - Called from Mailbox Controller code. Used here * to request the interrupt. @@ -490,6 +574,7 @@ static const struct mbox_chan_ops pcc_chan_ops = { .send_data = pcc_send_data, .startup = pcc_startup, .shutdown = pcc_shutdown, + .last_tx_done = pcc_last_tx_done, }; /** diff --git a/include/acpi/pcc.h b/include/acpi/pcc.h index 840bfc95bae3..b5414572912a 100644 --- a/include/acpi/pcc.h +++ b/include/acpi/pcc.h @@ -17,6 +17,25 @@ struct pcc_mbox_chan { u32 latency; u32 max_access_rate; u16 min_turnaround_time; + + /* Set to true When a message is sent that has the flag + * set that the client requests an Interrupt + * indicating that the transmission is complete. + */ + bool irq_ack; + /* Optional callback that allows the driver + * to allocate the memory used for receiving + * messages. The return value is the location + * inside the buffer where the mailbox should write the data. + */ + void *(*rx_alloc)(struct mbox_client *cl, int size); +}; + +struct pcc_header { + u32 signature; + u32 flags; + u32 length; + u32 command; }; /* Generic Communications Channel Shared Memory Region */ -- 2.43.0 ^ permalink raw reply related [flat|nested] 10+ messages in thread
* Re: [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer 2025-07-10 19:12 ` [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer admiyo @ 2025-07-11 6:51 ` kernel test robot 2025-07-14 19:37 ` Dan Carpenter 2025-07-14 23:02 ` Adam Young 2 siblings, 0 replies; 10+ messages in thread From: kernel test robot @ 2025-07-11 6:51 UTC (permalink / raw) To: admiyo, Sudeep Holla, Jassi Brar, Robert Moore, Rafael J. Wysocki, Len Brown Cc: llvm, oe-kbuild-all, netdev, linux-kernel, Jeremy Kerr, Matt Johnston, David S . Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni, Jonathan Cameron, Huisong Li Hi, kernel test robot noticed the following build warnings: [auto build test WARNING on net-next/main] url: https://github.com/intel-lab-lkp/linux/commits/admiyo-os-amperecomputing-com/mailbox-pcc-support-mailbox-management-of-the-shared-buffer/20250711-031525 base: net-next/main patch link: https://lore.kernel.org/r/20250710191209.737167-2-admiyo%40os.amperecomputing.com patch subject: [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer config: i386-randconfig-014-20250711 (https://download.01.org/0day-ci/archive/20250711/202507111440.1zQdhxpr-lkp@intel.com/config) compiler: clang version 20.1.8 (https://github.com/llvm/llvm-project 87f0227cb60147a26a1eeb4fb06e3b505e9c7261) reproduce (this is a W=1 build): (https://download.01.org/0day-ci/archive/20250711/202507111440.1zQdhxpr-lkp@intel.com/reproduce) If you fix the issue in a separate patch/commit (i.e. not just a new version of the same patch/commit), kindly add following tags | Reported-by: kernel test robot <lkp@intel.com> | Closes: https://lore.kernel.org/oe-kbuild-all/202507111440.1zQdhxpr-lkp@intel.com/ All warnings (new ones prefixed by >>): >> drivers/mailbox/pcc.c:496:6: warning: variable 'ret' is used uninitialized whenever 'if' condition is false [-Wsometimes-uninitialized] 496 | if (pchan->chan.rx_alloc) | ^~~~~~~~~~~~~~~~~~~~ drivers/mailbox/pcc.c:498:6: note: uninitialized use occurs here 498 | if (ret) | ^~~ drivers/mailbox/pcc.c:496:2: note: remove the 'if' if its condition is always true 496 | if (pchan->chan.rx_alloc) | ^~~~~~~~~~~~~~~~~~~~~~~~~ 497 | ret = pcc_write_to_buffer(chan, data); drivers/mailbox/pcc.c:492:9: note: initialize the variable 'ret' to silence this warning 492 | int ret; | ^ | = 0 1 warning generated. vim +496 drivers/mailbox/pcc.c 476 477 478 /** 479 * pcc_send_data - Called from Mailbox Controller code. Used 480 * here only to ring the channel doorbell. The PCC client 481 * specific read/write is done in the client driver in 482 * order to maintain atomicity over PCC channel once 483 * OS has control over it. See above for flow of operations. 484 * @chan: Pointer to Mailbox channel over which to send data. 485 * @data: Client specific data written over channel. Used here 486 * only for debug after PCC transaction completes. 487 * 488 * Return: Err if something failed else 0 for success. 489 */ 490 static int pcc_send_data(struct mbox_chan *chan, void *data) 491 { 492 int ret; 493 struct pcc_chan_info *pchan = chan->con_priv; 494 struct acpi_pcct_ext_pcc_shared_memory __iomem *pcc_hdr; 495 > 496 if (pchan->chan.rx_alloc) 497 ret = pcc_write_to_buffer(chan, data); 498 if (ret) 499 return ret; 500 501 ret = pcc_chan_reg_read_modify_write(&pchan->cmd_update); 502 if (ret) 503 return ret; 504 505 pcc_hdr = pchan->chan.shmem; 506 if (ioread32(&pcc_hdr->flags) & PCC_CMD_COMPLETION_NOTIFY) 507 pchan->chan.irq_ack = true; 508 509 ret = pcc_chan_reg_read_modify_write(&pchan->db); 510 511 if (!ret && pchan->plat_irq > 0) 512 pchan->chan_in_use = true; 513 514 return ret; 515 } 516 -- 0-DAY CI Kernel Test Service https://github.com/intel/lkp-tests/wiki ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer 2025-07-10 19:12 ` [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer admiyo 2025-07-11 6:51 ` kernel test robot @ 2025-07-14 19:37 ` Dan Carpenter 2025-07-14 23:02 ` Adam Young 2 siblings, 0 replies; 10+ messages in thread From: Dan Carpenter @ 2025-07-14 19:37 UTC (permalink / raw) To: oe-kbuild, admiyo, Sudeep Holla, Jassi Brar, Robert Moore, Rafael J. Wysocki, Len Brown Cc: lkp, oe-kbuild-all, netdev, linux-kernel, Jeremy Kerr, Matt Johnston, David S . Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni, Jonathan Cameron, Huisong Li Hi, kernel test robot noticed the following build warnings: url: https://github.com/intel-lab-lkp/linux/commits/admiyo-os-amperecomputing-com/mailbox-pcc-support-mailbox-management-of-the-shared-buffer/20250711-031525 base: net-next/main patch link: https://lore.kernel.org/r/20250710191209.737167-2-admiyo%40os.amperecomputing.com patch subject: [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer config: x86_64-randconfig-161-20250711 (https://download.01.org/0day-ci/archive/20250712/202507120609.Myazax08-lkp@intel.com/config) compiler: clang version 20.1.8 (https://github.com/llvm/llvm-project 87f0227cb60147a26a1eeb4fb06e3b505e9c7261) If you fix the issue in a separate patch/commit (i.e. not just a new version of the same patch/commit), kindly add following tags | Reported-by: kernel test robot <lkp@intel.com> | Reported-by: Dan Carpenter <dan.carpenter@linaro.org> | Closes: https://lore.kernel.org/r/202507120609.Myazax08-lkp@intel.com/ smatch warnings: drivers/mailbox/pcc.c:498 pcc_send_data() error: uninitialized symbol 'ret'. vim +/ret +498 drivers/mailbox/pcc.c 86c22f8c9a3b71 Ashwin Chaugule 2014-11-12 490 static int pcc_send_data(struct mbox_chan *chan, void *data) 86c22f8c9a3b71 Ashwin Chaugule 2014-11-12 491 { c45ded7e11352d Sudeep Holla 2021-09-17 492 int ret; bf18123e78f4d1 Sudeep Holla 2021-09-17 493 struct pcc_chan_info *pchan = chan->con_priv; e332edef98ddac Adam Young 2025-07-10 494 struct acpi_pcct_ext_pcc_shared_memory __iomem *pcc_hdr; e332edef98ddac Adam Young 2025-07-10 495 e332edef98ddac Adam Young 2025-07-10 496 if (pchan->chan.rx_alloc) e332edef98ddac Adam Young 2025-07-10 497 ret = pcc_write_to_buffer(chan, data); Hi Adam! :) ret is uninitialized on the else path. e332edef98ddac Adam Young 2025-07-10 @498 if (ret) e332edef98ddac Adam Young 2025-07-10 499 return ret; 8b0f57889843af Prakash, Prashanth 2016-02-17 500 c45ded7e11352d Sudeep Holla 2021-09-17 501 ret = pcc_chan_reg_read_modify_write(&pchan->cmd_update); c45ded7e11352d Sudeep Holla 2021-09-17 502 if (ret) c45ded7e11352d Sudeep Holla 2021-09-17 503 return ret; c45ded7e11352d Sudeep Holla 2021-09-17 504 e332edef98ddac Adam Young 2025-07-10 505 pcc_hdr = pchan->chan.shmem; e332edef98ddac Adam Young 2025-07-10 506 if (ioread32(&pcc_hdr->flags) & PCC_CMD_COMPLETION_NOTIFY) e332edef98ddac Adam Young 2025-07-10 507 pchan->chan.irq_ack = true; e332edef98ddac Adam Young 2025-07-10 508 3db174e478cb0b Huisong Li 2023-08-01 509 ret = pcc_chan_reg_read_modify_write(&pchan->db); e332edef98ddac Adam Young 2025-07-10 510 3db174e478cb0b Huisong Li 2023-08-01 511 if (!ret && pchan->plat_irq > 0) 3db174e478cb0b Huisong Li 2023-08-01 512 pchan->chan_in_use = true; 3db174e478cb0b Huisong Li 2023-08-01 513 3db174e478cb0b Huisong Li 2023-08-01 514 return ret; 86c22f8c9a3b71 Ashwin Chaugule 2014-11-12 515 } -- 0-DAY CI Kernel Test Service https://github.com/intel/lkp-tests/wiki ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer 2025-07-10 19:12 ` [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer admiyo 2025-07-11 6:51 ` kernel test robot 2025-07-14 19:37 ` Dan Carpenter @ 2025-07-14 23:02 ` Adam Young 2 siblings, 0 replies; 10+ messages in thread From: Adam Young @ 2025-07-14 23:02 UTC (permalink / raw) To: admiyo, Sudeep Holla, Jassi Brar, Robert Moore, Rafael J. Wysocki, Len Brown Cc: netdev, linux-kernel, Jeremy Kerr, Matt Johnston, David S . Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni, Jonathan Cameron, Huisong Li Internal review discovered an error On 7/10/25 15:12, admiyo@os.amperecomputing.com wrote: > + if (pchan->chan.rx_alloc) > + ret = pcc_write_to_buffer(chan, data); This is the wrong check. The rx_alloc is expected to be used for the type4, and the will be called from the type 3. Going to add an explicit flag instead. > + pcc_hdr = pchan->chan.shmem; > + if (ioread32(&pcc_hdr->flags) & PCC_CMD_COMPLETION_NOTIFY) > + pchan->chan.irq_ack = true; This flag can be removed and replaced with a check of the value in the original buffer, which is held in chan->current_req. Updated version of the patch series to follow. ^ permalink raw reply [flat|nested] 10+ messages in thread
* [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport 2025-07-10 19:12 [PATCH net-next v22 0/2] MCTP Over PCC Transport admiyo 2025-07-10 19:12 ` [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer admiyo @ 2025-07-10 19:12 ` admiyo 2025-07-11 11:02 ` kernel test robot 2025-07-11 12:52 ` Jeremy Kerr 1 sibling, 2 replies; 10+ messages in thread From: admiyo @ 2025-07-10 19:12 UTC (permalink / raw) To: Jeremy Kerr, Matt Johnston, Andrew Lunn, David S. Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni, Adam Young Cc: netdev, linux-kernel, Sudeep Holla, Jonathan Cameron, Huisong Li From: Adam Young <admiyo@os.amperecomputing.com> Implementation of network driver for Management Control Transport Protocol(MCTP) over Platform Communication Channel(PCC) DMTF DSP:0292 https://www.dmtf.org/sites/default/files/standards/documents/\ DSP0292_1.0.0WIP50.pdf MCTP devices are specified via ACPI by entries in DSDT/SDST and reference channels specified in the PCCT. Messages are sent on a type 3 and received on a type 4 channel. Communication with other devices use the PCC based doorbell mechanism; a shared memory segment with a corresponding interrupt and a memory register used to trigger remote interrupts. This driver takes advantage of PCC mailbox buffer management. The data section of the struct sk_buff that contains the outgoing packet is sent to the mailbox, already properly formatted as a PCC message. The driver is also reponsible for allocating a struct sk_buff that is then passed to the mailbox and used to record the data in the shared buffer. It maintains a list of both outging and incoming sk_buffs to match the data buffers with the original sk_buffs. When the Type 3 channel outbox receives a txdone response intterupt, it consumes the outgoing sk_buff, allowing it to be freed. Signed-off-by: Adam Young <admiyo@os.amperecomputing.com> --- MAINTAINERS | 5 + drivers/net/mctp/Kconfig | 13 ++ drivers/net/mctp/Makefile | 1 + drivers/net/mctp/mctp-pcc.c | 346 ++++++++++++++++++++++++++++++++++++ 4 files changed, 365 insertions(+) create mode 100644 drivers/net/mctp/mctp-pcc.c diff --git a/MAINTAINERS b/MAINTAINERS index 881a1f08e665..455618ee93b6 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -14467,6 +14467,11 @@ F: include/net/mctpdevice.h F: include/net/netns/mctp.h F: net/mctp/ +MANAGEMENT COMPONENT TRANSPORT PROTOCOL (MCTP) over PCC (MCTP-PCC) Driver +M: Adam Young <admiyo@os.amperecomputing.com> +S: Maintained +F: drivers/net/mctp/mctp-pcc.c + MAPLE TREE M: Liam R. Howlett <Liam.Howlett@oracle.com> L: maple-tree@lists.infradead.org diff --git a/drivers/net/mctp/Kconfig b/drivers/net/mctp/Kconfig index cf325ab0b1ef..f69d0237f058 100644 --- a/drivers/net/mctp/Kconfig +++ b/drivers/net/mctp/Kconfig @@ -57,6 +57,19 @@ config MCTP_TRANSPORT_USB MCTP-over-USB interfaces are peer-to-peer, so each interface represents a physical connection to one remote MCTP endpoint. +config MCTP_TRANSPORT_PCC + tristate "MCTP PCC transport" + depends on ACPI + help + Provides a driver to access MCTP devices over PCC transport, + A MCTP protocol network device is created via ACPI for each + entry in the DST/SDST that matches the identifier. The Platform + communication channels are selected from the corresponding + entries in the PCCT. + + Say y here if you need to connect to MCTP endpoints over PCC. To + compile as a module, use m; the module will be called mctp-pcc. + endmenu endif diff --git a/drivers/net/mctp/Makefile b/drivers/net/mctp/Makefile index c36006849a1e..2276f148df7c 100644 --- a/drivers/net/mctp/Makefile +++ b/drivers/net/mctp/Makefile @@ -1,3 +1,4 @@ +obj-$(CONFIG_MCTP_TRANSPORT_PCC) += mctp-pcc.o obj-$(CONFIG_MCTP_SERIAL) += mctp-serial.o obj-$(CONFIG_MCTP_TRANSPORT_I2C) += mctp-i2c.o obj-$(CONFIG_MCTP_TRANSPORT_I3C) += mctp-i3c.o diff --git a/drivers/net/mctp/mctp-pcc.c b/drivers/net/mctp/mctp-pcc.c new file mode 100644 index 000000000000..15ed449071d5 --- /dev/null +++ b/drivers/net/mctp/mctp-pcc.c @@ -0,0 +1,346 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * mctp-pcc.c - Driver for MCTP over PCC. + * Copyright (c) 2024, Ampere Computing LLC + */ + +/* Implementation of MCTP over PCC DMTF Specification DSP0256 + * https://www.dmtf.org/sites/default/files/standards/documents/DSP0256_2.0.0WIP50.pdf + */ + +#include <linux/acpi.h> +#include <linux/if_arp.h> +#include <linux/init.h> +#include <linux/kernel.h> +#include <linux/module.h> +#include <linux/netdevice.h> +#include <linux/platform_device.h> +#include <linux/string.h> +#include <linux/skbuff.h> +#include <linux/hrtimer.h> + +#include <acpi/acpi_bus.h> +#include <acpi/acpi_drivers.h> +#include <acpi/acrestyp.h> +#include <acpi/actbl.h> +#include <net/mctp.h> +#include <net/mctpdevice.h> +#include <acpi/pcc.h> + +#include "../../mailbox/mailbox.h" + +#define MCTP_PAYLOAD_LENGTH 256 +#define MCTP_CMD_LENGTH 4 +#define MCTP_PCC_VERSION 0x1 /* DSP0292 a single version: 1 */ +#define MCTP_SIGNATURE "MCTP" +#define MCTP_SIGNATURE_LENGTH (sizeof(MCTP_SIGNATURE) - 1) +#define MCTP_MIN_MTU 68 +#define PCC_DWORD_TYPE 0x0c + +struct mctp_pcc_mailbox { + u32 index; + struct pcc_mbox_chan *chan; + struct mbox_client client; + struct sk_buff_head packets; +}; + +/* The netdev structure. One of these per PCC adapter. */ +struct mctp_pcc_ndev { + /* spinlock to serialize access to PCC outbox buffer and registers + * Note that what PCC calls registers are memory locations, not CPU + * Registers. They include the fields used to synchronize access + * between the OS and remote endpoints. + * + * Only the Outbox needs a spinlock, to prevent multiple + * sent packets triggering multiple attempts to over write + * the outbox. The Inbox buffer is controlled by the remote + * service and a spinlock would have no effect. + */ + spinlock_t lock; + struct mctp_dev mdev; + struct acpi_device *acpi_device; + struct mctp_pcc_mailbox inbox; + struct mctp_pcc_mailbox outbox; +}; + +static void *mctp_pcc_rx_alloc(struct mbox_client *c, int size) +{ + struct mctp_pcc_mailbox *box; + struct mctp_pcc_ndev *mctp_pcc_ndev; + struct sk_buff *skb; + void *skb_buf; + + box = container_of(c, struct mctp_pcc_mailbox, client); + mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); + if (size > mctp_pcc_ndev->mdev.dev->mtu) + return NULL; + mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); + skb = netdev_alloc_skb(mctp_pcc_ndev->mdev.dev, size); + if (!skb) + return NULL; + skb_buf = skb_put(skb, size); + skb->protocol = htons(ETH_P_MCTP); + + skb_queue_head(&box->packets, skb); + + return skb->data; +} + +static void mctp_pcc_client_rx_callback(struct mbox_client *c, void *buffer) +{ + struct mctp_pcc_ndev *mctp_pcc_ndev; + struct pcc_header pcc_header; + struct mctp_skb_cb *cb; + struct sk_buff *skb; + + mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); + if (!buffer) { + dev_dstats_rx_dropped(mctp_pcc_ndev->mdev.dev); + return; + } + + skb_queue_walk(&mctp_pcc_ndev->inbox.packets, skb) { + if (skb->data == buffer) { + skb_unlink(skb, &mctp_pcc_ndev->inbox.packets); + dev_dstats_rx_add(mctp_pcc_ndev->mdev.dev, skb->len); + skb_reset_mac_header(skb); + skb_pull(skb, sizeof(pcc_header)); + skb_reset_network_header(skb); + cb = __mctp_cb(skb); + cb->halen = 0; + netif_rx(skb); + return; + } + } + pr_warn("Unmatched packet in mctp-pcc inbox packet list"); +} + +static void mctp_pcc_tx_done(struct mbox_client *c, void *mssg, int r) +{ + struct mctp_pcc_mailbox *box; + struct sk_buff *skb; + + box = container_of(c, struct mctp_pcc_mailbox, client); + skb_queue_walk(&box->packets, skb) { + if (skb->data == mssg) { + skb_unlink(skb, &box->packets); + dev_consume_skb_any(skb); + break; + } + } +} + +static netdev_tx_t mctp_pcc_tx(struct sk_buff *skb, struct net_device *ndev) +{ + struct mctp_pcc_ndev *mpnd = netdev_priv(ndev); + struct pcc_header *pcc_header; + int len = skb->len; + int rc; + + rc = skb_cow_head(skb, sizeof(*pcc_header)); + if (rc) + goto err_drop; + + pcc_header = skb_push(skb, sizeof(*pcc_header)); + pcc_header->signature = cpu_to_le32(PCC_SIGNATURE | mpnd->outbox.index); + pcc_header->flags = cpu_to_le32(PCC_CMD_COMPLETION_NOTIFY); + memcpy(&pcc_header->command, MCTP_SIGNATURE, MCTP_SIGNATURE_LENGTH); + pcc_header->length = cpu_to_le32(len + MCTP_SIGNATURE_LENGTH); + + skb_queue_head(&mpnd->outbox.packets, skb); + + rc = mbox_send_message(mpnd->outbox.chan->mchan, skb->data); + + if (rc < 0) { + pr_info("%s fail, rc = %d", __func__, rc); + return NETDEV_TX_BUSY; + } + dev_dstats_tx_add(ndev, len); + return NETDEV_TX_OK; +err_drop: + dev_dstats_tx_dropped(ndev); + kfree_skb(skb); + return NETDEV_TX_OK; +} + +static const struct net_device_ops mctp_pcc_netdev_ops = { + .ndo_start_xmit = mctp_pcc_tx, +}; + +static const struct mctp_netdev_ops mctp_netdev_ops = { + NULL +}; + +static void mctp_pcc_setup(struct net_device *ndev) +{ + ndev->type = ARPHRD_MCTP; + ndev->hard_header_len = 0; + ndev->tx_queue_len = 0; + ndev->flags = IFF_NOARP; + ndev->netdev_ops = &mctp_pcc_netdev_ops; + ndev->needs_free_netdev = true; + ndev->pcpu_stat_type = NETDEV_PCPU_STAT_DSTATS; +} + +struct mctp_pcc_lookup_context { + int index; + u32 inbox_index; + u32 outbox_index; +}; + +static acpi_status lookup_pcct_indices(struct acpi_resource *ares, + void *context) +{ + struct mctp_pcc_lookup_context *luc = context; + struct acpi_resource_address32 *addr; + + if (ares->type != PCC_DWORD_TYPE) + return AE_OK; + + addr = ACPI_CAST_PTR(struct acpi_resource_address32, &ares->data); + switch (luc->index) { + case 0: + luc->outbox_index = addr[0].address.minimum; + break; + case 1: + luc->inbox_index = addr[0].address.minimum; + break; + } + luc->index++; + return AE_OK; +} + +static void drain_packets(struct sk_buff_head *list) +{ + struct sk_buff *skb; + + while (!skb_queue_empty(list)) { + skb = skb_dequeue(list); + dev_consume_skb_any(skb); + } +} + +static void mctp_cleanup_netdev(void *data) +{ + struct mctp_pcc_ndev *mctp_pcc_ndev; + struct net_device *ndev = data; + + mctp_pcc_ndev = netdev_priv(ndev); + drain_packets(&mctp_pcc_ndev->outbox.packets); + drain_packets(&mctp_pcc_ndev->inbox.packets); + + mctp_unregister_netdev(ndev); +} + +static void mctp_cleanup_channel(void *data) +{ + struct pcc_mbox_chan *chan = data; + + pcc_mbox_free_channel(chan); +} + +static int mctp_pcc_initialize_mailbox(struct device *dev, + struct mctp_pcc_mailbox *box, u32 index) +{ + box->index = index; + skb_queue_head_init(&box->packets); + box->chan = pcc_mbox_request_channel(&box->client, index); + box->chan->rx_alloc = mctp_pcc_rx_alloc; + + box->client.dev = dev; + if (IS_ERR(box->chan)) + return PTR_ERR(box->chan); + return devm_add_action_or_reset(dev, mctp_cleanup_channel, box->chan); +} + +static int mctp_pcc_driver_add(struct acpi_device *acpi_dev) +{ + struct mctp_pcc_lookup_context context = {0}; + struct mctp_pcc_ndev *mctp_pcc_ndev; + struct device *dev = &acpi_dev->dev; + struct net_device *ndev; + acpi_handle dev_handle; + acpi_status status; + int mctp_pcc_mtu; + char name[32]; + int rc; + + dev_dbg(dev, "Adding mctp_pcc device for HID %s\n", + acpi_device_hid(acpi_dev)); + dev_handle = acpi_device_handle(acpi_dev); + status = acpi_walk_resources(dev_handle, "_CRS", lookup_pcct_indices, + &context); + if (!ACPI_SUCCESS(status)) { + dev_err(dev, "FAILURE to lookup PCC indexes from CRS\n"); + return -EINVAL; + } + + /* inbox initialization */ + snprintf(name, sizeof(name), "mctpipcc%d", context.inbox_index); + ndev = alloc_netdev(sizeof(*mctp_pcc_ndev), name, NET_NAME_PREDICTABLE, + mctp_pcc_setup); + if (!ndev) + return -ENOMEM; + + mctp_pcc_ndev = netdev_priv(ndev); + spin_lock_init(&mctp_pcc_ndev->lock); + rc = mctp_pcc_initialize_mailbox(dev, &mctp_pcc_ndev->inbox, + context.inbox_index); + if (rc) + goto free_netdev; + mctp_pcc_ndev->inbox.client.rx_callback = mctp_pcc_client_rx_callback; + + /* outbox initialization */ + rc = mctp_pcc_initialize_mailbox(dev, &mctp_pcc_ndev->outbox, + context.outbox_index); + if (rc) + goto free_netdev; + + mctp_pcc_ndev->outbox.client.tx_done = mctp_pcc_tx_done; + mctp_pcc_ndev->acpi_device = acpi_dev; + mctp_pcc_ndev->mdev.dev = ndev; + acpi_dev->driver_data = mctp_pcc_ndev; + + /* There is no clean way to pass the MTU to the callback function + * used for registration, so set the values ahead of time. + */ + mctp_pcc_mtu = mctp_pcc_ndev->outbox.chan->shmem_size - + sizeof(struct pcc_header); + ndev->mtu = MCTP_MIN_MTU; + ndev->max_mtu = mctp_pcc_mtu; + ndev->min_mtu = MCTP_MIN_MTU; + + /* ndev needs to be freed before the iomemory (mapped above) gets + * unmapped, devm resources get freed in reverse to the order they + * are added. + */ + rc = mctp_register_netdev(ndev, &mctp_netdev_ops, MCTP_PHYS_BINDING_PCC); + if (rc) + goto free_netdev; + return devm_add_action_or_reset(dev, mctp_cleanup_netdev, ndev); +free_netdev: + free_netdev(ndev); + return rc; +} + +static const struct acpi_device_id mctp_pcc_device_ids[] = { + { "DMT0001" }, + {} +}; + +static struct acpi_driver mctp_pcc_driver = { + .name = "mctp_pcc", + .class = "Unknown", + .ids = mctp_pcc_device_ids, + .ops = { + .add = mctp_pcc_driver_add, + }, +}; + +module_acpi_driver(mctp_pcc_driver); + +MODULE_DEVICE_TABLE(acpi, mctp_pcc_device_ids); + +MODULE_DESCRIPTION("MCTP PCC ACPI device"); +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("Adam Young <admiyo@os.amperecomputing.com>"); -- 2.43.0 ^ permalink raw reply related [flat|nested] 10+ messages in thread
* Re: [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport 2025-07-10 19:12 ` [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport admiyo @ 2025-07-11 11:02 ` kernel test robot 2025-07-11 12:52 ` Jeremy Kerr 1 sibling, 0 replies; 10+ messages in thread From: kernel test robot @ 2025-07-11 11:02 UTC (permalink / raw) To: admiyo, Jeremy Kerr, Matt Johnston, Andrew Lunn, David S. Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni Cc: oe-kbuild-all, netdev, linux-kernel, Sudeep Holla, Jonathan Cameron, Huisong Li Hi, kernel test robot noticed the following build warnings: [auto build test WARNING on net-next/main] url: https://github.com/intel-lab-lkp/linux/commits/admiyo-os-amperecomputing-com/mailbox-pcc-support-mailbox-management-of-the-shared-buffer/20250711-031525 base: net-next/main patch link: https://lore.kernel.org/r/20250710191209.737167-3-admiyo%40os.amperecomputing.com patch subject: [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport config: x86_64-allyesconfig (https://download.01.org/0day-ci/archive/20250711/202507111843.OV6uA3Jr-lkp@intel.com/config) compiler: clang version 20.1.8 (https://github.com/llvm/llvm-project 87f0227cb60147a26a1eeb4fb06e3b505e9c7261) reproduce (this is a W=1 build): (https://download.01.org/0day-ci/archive/20250711/202507111843.OV6uA3Jr-lkp@intel.com/reproduce) If you fix the issue in a separate patch/commit (i.e. not just a new version of the same patch/commit), kindly add following tags | Reported-by: kernel test robot <lkp@intel.com> | Closes: https://lore.kernel.org/oe-kbuild-all/202507111843.OV6uA3Jr-lkp@intel.com/ All warnings (new ones prefixed by >>): >> drivers/net/mctp/mctp-pcc.c:71:8: warning: variable 'skb_buf' set but not used [-Wunused-but-set-variable] 71 | void *skb_buf; | ^ 1 warning generated. vim +/skb_buf +71 drivers/net/mctp/mctp-pcc.c 65 66 static void *mctp_pcc_rx_alloc(struct mbox_client *c, int size) 67 { 68 struct mctp_pcc_mailbox *box; 69 struct mctp_pcc_ndev *mctp_pcc_ndev; 70 struct sk_buff *skb; > 71 void *skb_buf; 72 73 box = container_of(c, struct mctp_pcc_mailbox, client); 74 mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); 75 if (size > mctp_pcc_ndev->mdev.dev->mtu) 76 return NULL; 77 mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); 78 skb = netdev_alloc_skb(mctp_pcc_ndev->mdev.dev, size); 79 if (!skb) 80 return NULL; 81 skb_buf = skb_put(skb, size); 82 skb->protocol = htons(ETH_P_MCTP); 83 84 skb_queue_head(&box->packets, skb); 85 86 return skb->data; 87 } 88 -- 0-DAY CI Kernel Test Service https://github.com/intel/lkp-tests/wiki ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport 2025-07-10 19:12 ` [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport admiyo 2025-07-11 11:02 ` kernel test robot @ 2025-07-11 12:52 ` Jeremy Kerr 2025-07-11 20:03 ` Adam Young 1 sibling, 1 reply; 10+ messages in thread From: Jeremy Kerr @ 2025-07-11 12:52 UTC (permalink / raw) To: admiyo, Matt Johnston, Andrew Lunn, David S. Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni Cc: netdev, linux-kernel, Sudeep Holla, Jonathan Cameron, Huisong Li Hi Adam, A few comments inline: > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * mctp-pcc.c - Driver for MCTP over PCC. > + * Copyright (c) 2024, Ampere Computing LLC It's 2025 now, I'd suggest a range: Copyright (c) 2024-2025, Ampere Computing LLC > + */ > + > +/* Implementation of MCTP over PCC DMTF Specification DSP0256 > + * https://www.dmtf.org/sites/default/files/standards/documents/DSP0256_2.0.0WIP50.pdf DSP0256 has been released now, if that's a more appropriate reference. But it looks like DSP0292 is more specific to the PCC parts? > + */ > + > +#include <linux/acpi.h> > +#include <linux/if_arp.h> > +#include <linux/init.h> > +#include <linux/kernel.h> > +#include <linux/module.h> > +#include <linux/netdevice.h> > +#include <linux/platform_device.h> > +#include <linux/string.h> > +#include <linux/skbuff.h> > +#include <linux/hrtimer.h> > + > +#include <acpi/acpi_bus.h> > +#include <acpi/acpi_drivers.h> > +#include <acpi/acrestyp.h> > +#include <acpi/actbl.h> > +#include <net/mctp.h> > +#include <net/mctpdevice.h> > +#include <acpi/pcc.h> > + > +#include "../../mailbox/mailbox.h" > + > +#define MCTP_PAYLOAD_LENGTH 256 > +#define MCTP_CMD_LENGTH 4 > +#define MCTP_PCC_VERSION 0x1 /* DSP0292 a single version: 1 */ > +#define MCTP_SIGNATURE "MCTP" > +#define MCTP_SIGNATURE_LENGTH (sizeof(MCTP_SIGNATURE) - 1) > +#define MCTP_MIN_MTU 68 > +#define PCC_DWORD_TYPE 0x0c > + > +struct mctp_pcc_mailbox { > + u32 index; > + struct pcc_mbox_chan *chan; > + struct mbox_client client; > + struct sk_buff_head packets; > +}; > + > +/* The netdev structure. One of these per PCC adapter. */ > +struct mctp_pcc_ndev { > + /* spinlock to serialize access to PCC outbox buffer and registers > + * Note that what PCC calls registers are memory locations, not CPU > + * Registers. They include the fields used to synchronize access > + * between the OS and remote endpoints. > + * > + * Only the Outbox needs a spinlock, to prevent multiple > + * sent packets triggering multiple attempts to over write > + * the outbox. The Inbox buffer is controlled by the remote > + * service and a spinlock would have no effect. > + */ > + spinlock_t lock; > + struct mctp_dev mdev; You are only ever using the mdev->dev pointer; just use a struct net_device * here. The MCTP layer handles the creation of the MCTP parts. > + struct acpi_device *acpi_device; > + struct mctp_pcc_mailbox inbox; > + struct mctp_pcc_mailbox outbox; > +}; > + > +static void *mctp_pcc_rx_alloc(struct mbox_client *c, int size) > +{ > + struct mctp_pcc_mailbox *box; > + struct mctp_pcc_ndev *mctp_pcc_ndev; > + struct sk_buff *skb; > + void *skb_buf; > + > + box = container_of(c, struct mctp_pcc_mailbox, client); > + mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); > + if (size > mctp_pcc_ndev->mdev.dev->mtu) > + return NULL; > + mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); You already have set mctp_ncc_ndev a few lines above. It's also a bit unusual doing the two container_of() operations, when you can find the ndev and reference the mailbox from there. The common pattern for this is to set up your context from the input pointers early too, so something like: static void *mctp_pcc_rx_alloc(struct mbox_client *c, int size) { struct mctp_pcc_ndev *mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); struct mctp_pcc_mailbox *box = &mctp_pcc_ndev->inbox; struct sk_buff *skb; /* ... */ (but you may not need a var for 'box' at all) > + skb = netdev_alloc_skb(mctp_pcc_ndev->mdev.dev, size); > + if (!skb) > + return NULL; > + skb_buf = skb_put(skb, size); you don't use skb_buf anywhere? (building with W=1 should catch this) > + skb->protocol = htons(ETH_P_MCTP); > + > + skb_queue_head(&box->packets, skb); > + > + return skb->data; > +} > + > +static void mctp_pcc_client_rx_callback(struct mbox_client *c, void *buffer) > +{ > + struct mctp_pcc_ndev *mctp_pcc_ndev; > + struct pcc_header pcc_header; > + struct mctp_skb_cb *cb; > + struct sk_buff *skb; > + > + mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); > + if (!buffer) { > + dev_dstats_rx_dropped(mctp_pcc_ndev->mdev.dev); > + return; > + } Mainly out of curiosity: how does this happen? How do we get a completion where there is no original buffer? > + > + skb_queue_walk(&mctp_pcc_ndev->inbox.packets, skb) { > + if (skb->data == buffer) { > + skb_unlink(skb, &mctp_pcc_ndev->inbox.packets); > + dev_dstats_rx_add(mctp_pcc_ndev->mdev.dev, skb->len); > + skb_reset_mac_header(skb); > + skb_pull(skb, sizeof(pcc_header)); > + skb_reset_network_header(skb); > + cb = __mctp_cb(skb); > + cb->halen = 0; > + netif_rx(skb); > + return; > + } You can save a bit of indent by flipping the logic here: skb_queue_walk(&mctp_pcc_ndev->inbox.packets, skb) { if (skb->data != buffer) continue; skb_unlink(skb, &mctp_pcc_ndev->inbox.packets); dev_dstats_rx_add(mctp_pcc_ndev->mdev.dev, skb->len); skb_reset_mac_header(skb); skb_pull(skb, sizeof(pcc_header)); skb_reset_network_header(skb); cb = __mctp_cb(skb); cb->halen = 0; netif_rx(skb); return; } I figure we're restricted to what the mailbox API provides, but is there any way we can access the skb through a pointer, rather than having to dig through these lists? I think the issue is that the mbox API is using the void * buffer as both the data to transfer, and the callback context, so we can't stash useful context across the completion? > + } > + pr_warn("Unmatched packet in mctp-pcc inbox packet list"); > +} > + > +static void mctp_pcc_tx_done(struct mbox_client *c, void *mssg, int r) > +{ > + struct mctp_pcc_mailbox *box; > + struct sk_buff *skb; > + > + box = container_of(c, struct mctp_pcc_mailbox, client); > + skb_queue_walk(&box->packets, skb) { > + if (skb->data == mssg) { > + skb_unlink(skb, &box->packets); > + dev_consume_skb_any(skb); > + break; > + } > + } > +} > + > +static netdev_tx_t mctp_pcc_tx(struct sk_buff *skb, struct net_device *ndev) > +{ > + struct mctp_pcc_ndev *mpnd = netdev_priv(ndev); > + struct pcc_header *pcc_header; > + int len = skb->len; > + int rc; > + > + rc = skb_cow_head(skb, sizeof(*pcc_header)); > + if (rc) > + goto err_drop; > + > + pcc_header = skb_push(skb, sizeof(*pcc_header)); > + pcc_header->signature = cpu_to_le32(PCC_SIGNATURE | mpnd->outbox.index); > + pcc_header->flags = cpu_to_le32(PCC_CMD_COMPLETION_NOTIFY); > + memcpy(&pcc_header->command, MCTP_SIGNATURE, MCTP_SIGNATURE_LENGTH); > + pcc_header->length = cpu_to_le32(len + MCTP_SIGNATURE_LENGTH); > + > + skb_queue_head(&mpnd->outbox.packets, skb); > + > + rc = mbox_send_message(mpnd->outbox.chan->mchan, skb->data); > + > + if (rc < 0) { > + pr_info("%s fail, rc = %d", __func__, rc); > + return NETDEV_TX_BUSY; > + } What happens on mbox_send_message failure? The skb will still be present in the outbox.packets queue - I assume we don't see a completion callback in that case, and so the skb will be in the outbox.packets queue forever? Are you sure you want to return NETDEV_TX_BUSY here? Is there any situation where the mbox_send_message will continually fail? Should we ratelimit the pr_info() message there (and regardless, better to use one of netdev_info / netdev_warn / etc functions, since we are dealing with netdevs here). > + dev_dstats_tx_add(ndev, len); > + return NETDEV_TX_OK; > +err_drop: > + dev_dstats_tx_dropped(ndev); > + kfree_skb(skb); > + return NETDEV_TX_OK; > +} > + > +static const struct net_device_ops mctp_pcc_netdev_ops = { > + .ndo_start_xmit = mctp_pcc_tx, > +}; > + > +static const struct mctp_netdev_ops mctp_netdev_ops = { > + NULL > +}; > + > +static void mctp_pcc_setup(struct net_device *ndev) > +{ > + ndev->type = ARPHRD_MCTP; > + ndev->hard_header_len = 0; > + ndev->tx_queue_len = 0; > + ndev->flags = IFF_NOARP; > + ndev->netdev_ops = &mctp_pcc_netdev_ops; > + ndev->needs_free_netdev = true; > + ndev->pcpu_stat_type = NETDEV_PCPU_STAT_DSTATS; > +} > + > +struct mctp_pcc_lookup_context { > + int index; > + u32 inbox_index; > + u32 outbox_index; > +}; > + > +static acpi_status lookup_pcct_indices(struct acpi_resource *ares, > + void *context) > +{ > + struct mctp_pcc_lookup_context *luc = context; > + struct acpi_resource_address32 *addr; > + > + if (ares->type != PCC_DWORD_TYPE) > + return AE_OK; > + > + addr = ACPI_CAST_PTR(struct acpi_resource_address32, &ares->data); > + switch (luc->index) { > + case 0: > + luc->outbox_index = addr[0].address.minimum; > + break; > + case 1: > + luc->inbox_index = addr[0].address.minimum; > + break; > + } > + luc->index++; > + return AE_OK; > +} > + > +static void drain_packets(struct sk_buff_head *list) > +{ > + struct sk_buff *skb; > + > + while (!skb_queue_empty(list)) { > + skb = skb_dequeue(list); > + dev_consume_skb_any(skb); > + } > +} > + > +static void mctp_cleanup_netdev(void *data) > +{ > + struct mctp_pcc_ndev *mctp_pcc_ndev; > + struct net_device *ndev = data; > + > + mctp_pcc_ndev = netdev_priv(ndev); > + drain_packets(&mctp_pcc_ndev->outbox.packets); > + drain_packets(&mctp_pcc_ndev->inbox.packets); > + > + mctp_unregister_netdev(ndev); > +} > + > +static void mctp_cleanup_channel(void *data) > +{ > + struct pcc_mbox_chan *chan = data; > + > + pcc_mbox_free_channel(chan); > +} > + > +static int mctp_pcc_initialize_mailbox(struct device *dev, > + struct mctp_pcc_mailbox *box, u32 index) > +{ > + box->index = index; > + skb_queue_head_init(&box->packets); > + box->chan = pcc_mbox_request_channel(&box->client, index); > + box->chan->rx_alloc = mctp_pcc_rx_alloc; > + > + box->client.dev = dev; > + if (IS_ERR(box->chan)) > + return PTR_ERR(box->chan); > + return devm_add_action_or_reset(dev, mctp_cleanup_channel, box->chan); > +} > + > +static int mctp_pcc_driver_add(struct acpi_device *acpi_dev) > +{ > + struct mctp_pcc_lookup_context context = {0}; > + struct mctp_pcc_ndev *mctp_pcc_ndev; > + struct device *dev = &acpi_dev->dev; > + struct net_device *ndev; > + acpi_handle dev_handle; > + acpi_status status; > + int mctp_pcc_mtu; > + char name[32]; > + int rc; > + > + dev_dbg(dev, "Adding mctp_pcc device for HID %s\n", > + acpi_device_hid(acpi_dev)); > + dev_handle = acpi_device_handle(acpi_dev); > + status = acpi_walk_resources(dev_handle, "_CRS", lookup_pcct_indices, > + &context); > + if (!ACPI_SUCCESS(status)) { > + dev_err(dev, "FAILURE to lookup PCC indexes from CRS\n"); > + return -EINVAL; > + } > + > + /* inbox initialization */ the inbox initialization seems to be a bit further down. > + snprintf(name, sizeof(name), "mctpipcc%d", context.inbox_index); > + ndev = alloc_netdev(sizeof(*mctp_pcc_ndev), name, NET_NAME_PREDICTABLE, > + mctp_pcc_setup); > + if (!ndev) > + return -ENOMEM; > + > + mctp_pcc_ndev = netdev_priv(ndev); > + spin_lock_init(&mctp_pcc_ndev->lock); > + rc = mctp_pcc_initialize_mailbox(dev, &mctp_pcc_ndev->inbox, > + context.inbox_index); > + if (rc) > + goto free_netdev; > + mctp_pcc_ndev->inbox.client.rx_callback = mctp_pcc_client_rx_callback; > + > + /* outbox initialization */ > + rc = mctp_pcc_initialize_mailbox(dev, &mctp_pcc_ndev->outbox, > + context.outbox_index); > + if (rc) > + goto free_netdev; > + > + mctp_pcc_ndev->outbox.client.tx_done = mctp_pcc_tx_done; > + mctp_pcc_ndev->acpi_device = acpi_dev; > + mctp_pcc_ndev->mdev.dev = ndev; > + acpi_dev->driver_data = mctp_pcc_ndev; > + > + /* There is no clean way to pass the MTU to the callback function > + * used for registration, so set the values ahead of time. > + */ > + mctp_pcc_mtu = mctp_pcc_ndev->outbox.chan->shmem_size - > + sizeof(struct pcc_header); > + ndev->mtu = MCTP_MIN_MTU; > + ndev->max_mtu = mctp_pcc_mtu; > + ndev->min_mtu = MCTP_MIN_MTU; > + > + /* ndev needs to be freed before the iomemory (mapped above) gets > + * unmapped, devm resources get freed in reverse to the order they > + * are added. > + */ > + rc = mctp_register_netdev(ndev, &mctp_netdev_ops, MCTP_PHYS_BINDING_PCC); > + if (rc) > + goto free_netdev; > + return devm_add_action_or_reset(dev, mctp_cleanup_netdev, ndev); > +free_netdev: > + free_netdev(ndev); > + return rc; > +} Just a couple of nitpicky style things here (and above): try to keep the return-value checks immediately after the appropriate call: rc = mbox_send_message(mpnd->outbox.chan->mchan, skb->data); if (rc < 0) { pr_info("%s fail, rc = %d", __func__, rc); return NETDEV_TX_BUSY; } but give yourself some space after those checks, and around the returns/goto labels: rc = mctp_register_netdev(ndev, &mctp_netdev_ops, MCTP_PHYS_BINDING_PCC); if (rc) goto free_netdev; return devm_add_action_or_reset(dev, mctp_cleanup_netdev, ndev); free_netdev: free_netdev(ndev); return rc; } Cheers, Jeremy ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport 2025-07-11 12:52 ` Jeremy Kerr @ 2025-07-11 20:03 ` Adam Young 2025-07-14 8:21 ` Jeremy Kerr 0 siblings, 1 reply; 10+ messages in thread From: Adam Young @ 2025-07-11 20:03 UTC (permalink / raw) To: Jeremy Kerr, admiyo, Matt Johnston, Andrew Lunn, David S. Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni Cc: netdev, linux-kernel, Sudeep Holla, Jonathan Cameron, Huisong Li All Changes are accepted. I have attempted to answer your questions here. >> +static void mctp_pcc_client_rx_callback(struct mbox_client *c, void *buffer) >> +{ >> + struct mctp_pcc_ndev *mctp_pcc_ndev; >> + struct pcc_header pcc_header; >> + struct mctp_skb_cb *cb; >> + struct sk_buff *skb; >> + >> + mctp_pcc_ndev = container_of(c, struct mctp_pcc_ndev, inbox.client); >> + if (!buffer) { >> + dev_dstats_rx_dropped(mctp_pcc_ndev->mdev.dev); >> + return; >> + } > Mainly out of curiosity: how does this happen? How do we get a > completion where there is no original buffer? If the sk_buff allocation fails, the logic falls back to the old code, which passes on a null buffer. There is logic there with notifying the sender that I don't want to skip or modify. See the other patch, in the change to the irq handler. > I figure we're restricted to what the mailbox API provides, but is there > any way we can access the skb through a pointer, rather than having to > dig through these lists? > > I think the issue is that the mbox API is using the void * buffer as > both the data to transfer, and the callback context, so we can't stash > useful context across the completion? Correct, the SK_buff is a structure that points to a buffer, and what gets to the send_data function is the buffer itself. That buffer has no pointer back to the sk_buff. >> + >> + rc = mbox_send_message(mpnd->outbox.chan->mchan, skb->data); >> + >> + if (rc < 0) { >> + pr_info("%s fail, rc = %d", __func__, rc); >> + return NETDEV_TX_BUSY; >> + } > What happens on mbox_send_message failure? The skb will still be present > in the outbox.packets queue - I assume we don't see a completion > callback in that case, and so the skb will be in the outbox.packets > queue forever? > > Are you sure you want to return NETDEV_TX_BUSY here? > > Is there any situation where the mbox_send_message will continually > fail? Should we ratelimit the pr_info() message there (and regardless, > better to use one of netdev_info / netdev_warn / etc functions, since we > are dealing with netdevs here). The fail will happen if the ring buffer is full, so, yes, it makes sense not to queue the packet. I can move that to after the mbox_send_message. The NETDEV_TX_BUSY is correct, as it means resend the packet, and we don't have any reference to it. The only other failure path on the mbox_send_message code is due to timeouts, which we do not use from this driver. That is a recent change, and that was the case I was handling before. ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport 2025-07-11 20:03 ` Adam Young @ 2025-07-14 8:21 ` Jeremy Kerr 0 siblings, 0 replies; 10+ messages in thread From: Jeremy Kerr @ 2025-07-14 8:21 UTC (permalink / raw) To: Adam Young, admiyo, Matt Johnston, Andrew Lunn, David S. Miller, Eric Dumazet, Jakub Kicinski, Paolo Abeni Cc: netdev, linux-kernel, Sudeep Holla, Jonathan Cameron, Huisong Li Hi Adam, > If the sk_buff allocation fails, the logic falls back to the old code, > which passes on a null buffer. There is logic there with notifying the > sender that I don't want to skip or modify. OK, so this will happen if we didn't allocate a buffer in the first place - we'd still get a completion occurring. Let me know if my understanding is incorrect. > > > I think the issue is that the mbox API is using the void * buffer > > as both the data to transfer, and the callback context, so we can't > > stash useful context across the completion? > > Correct, the SK_buff is a structure that points to a buffer, and > what gets to the send_data function is the buffer itself. That buffer > has no pointer back to the sk_buff. OK, that's a bit unfortunate. Might be good to see if you can add a context pointer to the mailbox request, in which you could stuff the skb pointer. USB does this for the transfer completions: there are members on the transfer (the struct urb) for both a context and buffer pointer. Of course, that is more a mailbox API change, so you may want to consider that as a separate thing later. > The NETDEV_TX_BUSY is correct, as it means resend the packet, and we > don't have any reference to it. OK, you may want to have the tx queue stopped at that point then, if you have some facility to re-start it when the mailbox ring buffer has space. Cheers, Jeremy ^ permalink raw reply [flat|nested] 10+ messages in thread
end of thread, other threads:[~2025-07-14 23:02 UTC | newest] Thread overview: 10+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2025-07-10 19:12 [PATCH net-next v22 0/2] MCTP Over PCC Transport admiyo 2025-07-10 19:12 ` [PATCH net-next v22 1/2] mailbox/pcc: support mailbox management of the shared buffer admiyo 2025-07-11 6:51 ` kernel test robot 2025-07-14 19:37 ` Dan Carpenter 2025-07-14 23:02 ` Adam Young 2025-07-10 19:12 ` [PATCH net-next v22 2/2] mctp pcc: Implement MCTP over PCC Transport admiyo 2025-07-11 11:02 ` kernel test robot 2025-07-11 12:52 ` Jeremy Kerr 2025-07-11 20:03 ` Adam Young 2025-07-14 8:21 ` Jeremy Kerr
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox; as well as URLs for NNTP newsgroup(s).