/** * ntb_msi_init() - Initialize the MSI context * @ntb: NTB device context * * This function must be called before any other ntb_msi function. * It initializes the context for MSI operations and maps * the peer memory windows. * * This function reserves the last N outbound memory windows (where N * is the number of peers). * * Return: Zero on success, otherwise a negative error number.
*/ int ntb_msi_init(struct ntb_dev *ntb, void (*desc_changed)(void *ctx))
{
phys_addr_t mw_phys_addr;
resource_size_t mw_size; int peer_widx; int peers; int ret; int i;
peers = ntb_peer_port_count(ntb); if (peers <= 0) return -EINVAL;
/** * ntb_msi_setup_mws() - Initialize the MSI inbound memory windows * @ntb: NTB device context * * This function sets up the required inbound memory windows. It should be * called from a work function after a link up event. * * Over the entire network, this function will reserves the last N * inbound memory windows for each peer (where N is the number of peers). * * ntb_msi_init() must be called before this function. * * Return: Zero on success, otherwise a negative error number.
*/ int ntb_msi_setup_mws(struct ntb_dev *ntb)
{ struct msi_desc *desc;
u64 addr; int peer, peer_widx;
resource_size_t addr_align, size_align, size_max;
resource_size_t mw_size = SZ_32K;
resource_size_t mw_min_size = mw_size; int i; int ret;
error_out: for (i = 0; i < peer; i++) {
peer_widx = ntb_peer_highest_mw_idx(ntb, peer); if (peer_widx < 0) continue;
ntb_mw_clear_trans(ntb, i, peer_widx);
}
return ret;
}
EXPORT_SYMBOL(ntb_msi_setup_mws);
/** * ntb_msi_clear_mws() - Clear all inbound memory windows * @ntb: NTB device context * * This function tears down the resources used by ntb_msi_setup_mws().
*/ void ntb_msi_clear_mws(struct ntb_dev *ntb)
{ int peer; int peer_widx;
for (peer = 0; peer < ntb_peer_port_count(ntb); peer++) {
peer_widx = ntb_peer_highest_mw_idx(ntb, peer); if (peer_widx < 0) continue;
/** * ntbm_msi_request_threaded_irq() - allocate an MSI interrupt * @ntb: NTB device context * @handler: Function to be called when the IRQ occurs * @thread_fn: Function to be called in a threaded interrupt context. NULL * for clients which handle everything in @handler * @name: An ascii name for the claiming device, dev_name(dev) if NULL * @dev_id: A cookie passed back to the handler function * @msi_desc: MSI descriptor data which triggers the interrupt * * This function assigns an interrupt handler to an unused * MSI interrupt and returns the descriptor used to trigger * it. The descriptor can then be sent to a peer to trigger * the interrupt. * * The interrupt resource is managed with devres so it will * be automatically freed when the NTB device is torn down. * * If an IRQ allocated with this function needs to be freed * separately, ntbm_free_irq() must be used. * * Return: IRQ number assigned on success, otherwise a negative error number.
*/ int ntbm_msi_request_threaded_irq(struct ntb_dev *ntb, irq_handler_t handler,
irq_handler_t thread_fn, constchar *name, void *dev_id, struct ntb_msi_desc *msi_desc)
{ struct device *dev = &ntb->pdev->dev; struct msi_desc *entry; int ret;
if (!ntb->msi) return -EINVAL;
guard(msi_descs_lock)(dev);
msi_for_each_desc(entry, dev, MSI_DESC_ASSOCIATED) { if (irq_has_action(entry->irq)) continue;
ret = devm_request_threaded_irq(&ntb->dev, entry->irq, handler,
thread_fn, 0, name, dev_id); if (ret) continue;
if (ntb_msi_set_desc(ntb, entry, msi_desc)) {
devm_free_irq(&ntb->dev, entry->irq, dev_id); continue;
}
/** * ntbm_msi_free_irq() - free an interrupt * @ntb: NTB device context * @irq: Interrupt line to free * @dev_id: Device identity to free * * This function should be used to manually free IRQs allocated with * ntbm_request_[threaded_]irq().
*/ void ntbm_msi_free_irq(struct ntb_dev *ntb, unsignedint irq, void *dev_id)
{ struct msi_desc *entry = irq_get_msi_desc(irq);
/** * ntb_msi_peer_trigger() - Trigger an interrupt handler on a peer * @ntb: NTB device context * @peer: Peer index * @desc: MSI descriptor data which triggers the interrupt * * This function triggers an interrupt on a peer. It requires * the descriptor structure to have been passed from that peer * by some other means. * * Return: Zero on success, otherwise a negative error number.
*/ int ntb_msi_peer_trigger(struct ntb_dev *ntb, int peer, struct ntb_msi_desc *desc)
{ int idx;
/** * ntb_msi_peer_addr() - Get the DMA address to trigger a peer's MSI interrupt * @ntb: NTB device context * @peer: Peer index * @desc: MSI descriptor data which triggers the interrupt * @msi_addr: Physical address to trigger the interrupt * * This function allows using DMA engines to trigger an interrupt * (for example, trigger an interrupt to process the data after * sending it). To trigger the interrupt, write @desc.data to the address * returned in @msi_addr * * Return: Zero on success, otherwise a negative error number.
*/ int ntb_msi_peer_addr(struct ntb_dev *ntb, int peer, struct ntb_msi_desc *desc,
phys_addr_t *msi_addr)
{ int peer_widx = ntb_peer_mw_count(ntb) - 1 - peer;
phys_addr_t mw_phys_addr; int ret;
ret = ntb_peer_mw_get_addr(ntb, peer_widx, &mw_phys_addr, NULL); if (ret) return ret;
if (msi_addr)
*msi_addr = mw_phys_addr + desc->addr_offset;
return 0;
}
EXPORT_SYMBOL(ntb_msi_peer_addr);
Messung V0.5
¤ Dauer der Verarbeitung: 0.21 Sekunden
(vorverarbeitet)
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.