|  | /* | 
|  | * Copyright 2008 Advanced Micro Devices, Inc. | 
|  | * Copyright 2008 Red Hat Inc. | 
|  | * Copyright 2009 Jerome Glisse. | 
|  | * | 
|  | * Permission is hereby granted, free of charge, to any person obtaining a | 
|  | * copy of this software and associated documentation files (the "Software"), | 
|  | * to deal in the Software without restriction, including without limitation | 
|  | * the rights to use, copy, modify, merge, publish, distribute, sublicense, | 
|  | * and/or sell copies of the Software, and to permit persons to whom the | 
|  | * Software is furnished to do so, subject to the following conditions: | 
|  | * | 
|  | * The above copyright notice and this permission notice shall be included in | 
|  | * all copies or substantial portions of the Software. | 
|  | * | 
|  | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
|  | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
|  | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL | 
|  | * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR | 
|  | * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, | 
|  | * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR | 
|  | * OTHER DEALINGS IN THE SOFTWARE. | 
|  | * | 
|  | * Authors: Dave Airlie | 
|  | *          Alex Deucher | 
|  | *          Jerome Glisse | 
|  | */ | 
|  |  | 
|  | /** | 
|  | * DOC: Interrupt Handling | 
|  | * | 
|  | * Interrupts generated within GPU hardware raise interrupt requests that are | 
|  | * passed to amdgpu IRQ handler which is responsible for detecting source and | 
|  | * type of the interrupt and dispatching matching handlers. If handling an | 
|  | * interrupt requires calling kernel functions that may sleep processing is | 
|  | * dispatched to work handlers. | 
|  | * | 
|  | * If MSI functionality is not disabled by module parameter then MSI | 
|  | * support will be enabled. | 
|  | * | 
|  | * For GPU interrupt sources that may be driven by another driver, IRQ domain | 
|  | * support is used (with mapping between virtual and hardware IRQs). | 
|  | */ | 
|  |  | 
|  | #include <linux/irq.h> | 
|  | #include <linux/pci.h> | 
|  |  | 
|  | #include <drm/drm_crtc_helper.h> | 
|  | #include <drm/drm_vblank.h> | 
|  | #include <drm/amdgpu_drm.h> | 
|  | #include <drm/drm_drv.h> | 
|  | #include "amdgpu.h" | 
|  | #include "amdgpu_ih.h" | 
|  | #include "atom.h" | 
|  | #include "amdgpu_connectors.h" | 
|  | #include "amdgpu_trace.h" | 
|  | #include "amdgpu_amdkfd.h" | 
|  | #include "amdgpu_ras.h" | 
|  |  | 
|  | #include <linux/pm_runtime.h> | 
|  |  | 
|  | #ifdef CONFIG_DRM_AMD_DC | 
|  | #include "amdgpu_dm_irq.h" | 
|  | #endif | 
|  |  | 
|  | #define AMDGPU_WAIT_IDLE_TIMEOUT 200 | 
|  |  | 
|  | const char *soc15_ih_clientid_name[] = { | 
|  | "IH", | 
|  | "SDMA2 or ACP", | 
|  | "ATHUB", | 
|  | "BIF", | 
|  | "SDMA3 or DCE", | 
|  | "SDMA4 or ISP", | 
|  | "VMC1 or PCIE0", | 
|  | "RLC", | 
|  | "SDMA0", | 
|  | "SDMA1", | 
|  | "SE0SH", | 
|  | "SE1SH", | 
|  | "SE2SH", | 
|  | "SE3SH", | 
|  | "VCN1 or UVD1", | 
|  | "THM", | 
|  | "VCN or UVD", | 
|  | "SDMA5 or VCE0", | 
|  | "VMC", | 
|  | "SDMA6 or XDMA", | 
|  | "GRBM_CP", | 
|  | "ATS", | 
|  | "ROM_SMUIO", | 
|  | "DF", | 
|  | "SDMA7 or VCE1", | 
|  | "PWR", | 
|  | "reserved", | 
|  | "UTCL2", | 
|  | "EA", | 
|  | "UTCL2LOG", | 
|  | "MP0", | 
|  | "MP1" | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * amdgpu_hotplug_work_func - work handler for display hotplug event | 
|  | * | 
|  | * @work: work struct pointer | 
|  | * | 
|  | * This is the hotplug event work handler (all ASICs). | 
|  | * The work gets scheduled from the IRQ handler if there | 
|  | * was a hotplug interrupt.  It walks through the connector table | 
|  | * and calls hotplug handler for each connector. After this, it sends | 
|  | * a DRM hotplug event to alert userspace. | 
|  | * | 
|  | * This design approach is required in order to defer hotplug event handling | 
|  | * from the IRQ handler to a work handler because hotplug handler has to use | 
|  | * mutexes which cannot be locked in an IRQ handler (since &mutex_lock may | 
|  | * sleep). | 
|  | */ | 
|  | static void amdgpu_hotplug_work_func(struct work_struct *work) | 
|  | { | 
|  | struct amdgpu_device *adev = container_of(work, struct amdgpu_device, | 
|  | hotplug_work); | 
|  | struct drm_device *dev = adev_to_drm(adev); | 
|  | struct drm_mode_config *mode_config = &dev->mode_config; | 
|  | struct drm_connector *connector; | 
|  | struct drm_connector_list_iter iter; | 
|  |  | 
|  | mutex_lock(&mode_config->mutex); | 
|  | drm_connector_list_iter_begin(dev, &iter); | 
|  | drm_for_each_connector_iter(connector, &iter) | 
|  | amdgpu_connector_hotplug(connector); | 
|  | drm_connector_list_iter_end(&iter); | 
|  | mutex_unlock(&mode_config->mutex); | 
|  | /* Just fire off a uevent and let userspace tell us what to do */ | 
|  | drm_helper_hpd_irq_event(dev); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_disable_all - disable *all* interrupts | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * | 
|  | * Disable all types of interrupts from all sources. | 
|  | */ | 
|  | void amdgpu_irq_disable_all(struct amdgpu_device *adev) | 
|  | { | 
|  | unsigned long irqflags; | 
|  | unsigned i, j, k; | 
|  | int r; | 
|  |  | 
|  | spin_lock_irqsave(&adev->irq.lock, irqflags); | 
|  | for (i = 0; i < AMDGPU_IRQ_CLIENTID_MAX; ++i) { | 
|  | if (!adev->irq.client[i].sources) | 
|  | continue; | 
|  |  | 
|  | for (j = 0; j < AMDGPU_MAX_IRQ_SRC_ID; ++j) { | 
|  | struct amdgpu_irq_src *src = adev->irq.client[i].sources[j]; | 
|  |  | 
|  | if (!src || !src->funcs->set || !src->num_types) | 
|  | continue; | 
|  |  | 
|  | for (k = 0; k < src->num_types; ++k) { | 
|  | atomic_set(&src->enabled_types[k], 0); | 
|  | r = src->funcs->set(adev, src, k, | 
|  | AMDGPU_IRQ_STATE_DISABLE); | 
|  | if (r) | 
|  | DRM_ERROR("error disabling interrupt (%d)\n", | 
|  | r); | 
|  | } | 
|  | } | 
|  | } | 
|  | spin_unlock_irqrestore(&adev->irq.lock, irqflags); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_handler - IRQ handler | 
|  | * | 
|  | * @irq: IRQ number (unused) | 
|  | * @arg: pointer to DRM device | 
|  | * | 
|  | * IRQ handler for amdgpu driver (all ASICs). | 
|  | * | 
|  | * Returns: | 
|  | * result of handling the IRQ, as defined by &irqreturn_t | 
|  | */ | 
|  | static irqreturn_t amdgpu_irq_handler(int irq, void *arg) | 
|  | { | 
|  | struct drm_device *dev = (struct drm_device *) arg; | 
|  | struct amdgpu_device *adev = drm_to_adev(dev); | 
|  | irqreturn_t ret; | 
|  |  | 
|  | ret = amdgpu_ih_process(adev, &adev->irq.ih); | 
|  | if (ret == IRQ_HANDLED) | 
|  | pm_runtime_mark_last_busy(dev->dev); | 
|  |  | 
|  | /* For the hardware that cannot enable bif ring for both ras_controller_irq | 
|  | * and ras_err_evnet_athub_irq ih cookies, the driver has to poll status | 
|  | * register to check whether the interrupt is triggered or not, and properly | 
|  | * ack the interrupt if it is there | 
|  | */ | 
|  | if (amdgpu_ras_is_supported(adev, AMDGPU_RAS_BLOCK__PCIE_BIF)) { | 
|  | if (adev->nbio.ras_funcs && | 
|  | adev->nbio.ras_funcs->handle_ras_controller_intr_no_bifring) | 
|  | adev->nbio.ras_funcs->handle_ras_controller_intr_no_bifring(adev); | 
|  |  | 
|  | if (adev->nbio.ras_funcs && | 
|  | adev->nbio.ras_funcs->handle_ras_err_event_athub_intr_no_bifring) | 
|  | adev->nbio.ras_funcs->handle_ras_err_event_athub_intr_no_bifring(adev); | 
|  | } | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_handle_ih1 - kick of processing for IH1 | 
|  | * | 
|  | * @work: work structure in struct amdgpu_irq | 
|  | * | 
|  | * Kick of processing IH ring 1. | 
|  | */ | 
|  | static void amdgpu_irq_handle_ih1(struct work_struct *work) | 
|  | { | 
|  | struct amdgpu_device *adev = container_of(work, struct amdgpu_device, | 
|  | irq.ih1_work); | 
|  |  | 
|  | amdgpu_ih_process(adev, &adev->irq.ih1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_handle_ih2 - kick of processing for IH2 | 
|  | * | 
|  | * @work: work structure in struct amdgpu_irq | 
|  | * | 
|  | * Kick of processing IH ring 2. | 
|  | */ | 
|  | static void amdgpu_irq_handle_ih2(struct work_struct *work) | 
|  | { | 
|  | struct amdgpu_device *adev = container_of(work, struct amdgpu_device, | 
|  | irq.ih2_work); | 
|  |  | 
|  | amdgpu_ih_process(adev, &adev->irq.ih2); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_handle_ih_soft - kick of processing for ih_soft | 
|  | * | 
|  | * @work: work structure in struct amdgpu_irq | 
|  | * | 
|  | * Kick of processing IH soft ring. | 
|  | */ | 
|  | static void amdgpu_irq_handle_ih_soft(struct work_struct *work) | 
|  | { | 
|  | struct amdgpu_device *adev = container_of(work, struct amdgpu_device, | 
|  | irq.ih_soft_work); | 
|  |  | 
|  | amdgpu_ih_process(adev, &adev->irq.ih_soft); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_msi_ok - check whether MSI functionality is enabled | 
|  | * | 
|  | * @adev: amdgpu device pointer (unused) | 
|  | * | 
|  | * Checks whether MSI functionality has been disabled via module parameter | 
|  | * (all ASICs). | 
|  | * | 
|  | * Returns: | 
|  | * *true* if MSIs are allowed to be enabled or *false* otherwise | 
|  | */ | 
|  | static bool amdgpu_msi_ok(struct amdgpu_device *adev) | 
|  | { | 
|  | if (amdgpu_msi == 1) | 
|  | return true; | 
|  | else if (amdgpu_msi == 0) | 
|  | return false; | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static void amdgpu_restore_msix(struct amdgpu_device *adev) | 
|  | { | 
|  | u16 ctrl; | 
|  |  | 
|  | pci_read_config_word(adev->pdev, adev->pdev->msix_cap + PCI_MSIX_FLAGS, &ctrl); | 
|  | if (!(ctrl & PCI_MSIX_FLAGS_ENABLE)) | 
|  | return; | 
|  |  | 
|  | /* VF FLR */ | 
|  | ctrl &= ~PCI_MSIX_FLAGS_ENABLE; | 
|  | pci_write_config_word(adev->pdev, adev->pdev->msix_cap + PCI_MSIX_FLAGS, ctrl); | 
|  | ctrl |= PCI_MSIX_FLAGS_ENABLE; | 
|  | pci_write_config_word(adev->pdev, adev->pdev->msix_cap + PCI_MSIX_FLAGS, ctrl); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_init - initialize interrupt handling | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * | 
|  | * Sets up work functions for hotplug and reset interrupts, enables MSI | 
|  | * functionality, initializes vblank, hotplug and reset interrupt handling. | 
|  | * | 
|  | * Returns: | 
|  | * 0 on success or error code on failure | 
|  | */ | 
|  | int amdgpu_irq_init(struct amdgpu_device *adev) | 
|  | { | 
|  | int r = 0; | 
|  | unsigned int irq; | 
|  |  | 
|  | spin_lock_init(&adev->irq.lock); | 
|  |  | 
|  | /* Enable MSI if not disabled by module parameter */ | 
|  | adev->irq.msi_enabled = false; | 
|  |  | 
|  | if (amdgpu_msi_ok(adev)) { | 
|  | int nvec = pci_msix_vec_count(adev->pdev); | 
|  | unsigned int flags; | 
|  |  | 
|  | if (nvec <= 0) { | 
|  | flags = PCI_IRQ_MSI; | 
|  | } else { | 
|  | flags = PCI_IRQ_MSI | PCI_IRQ_MSIX; | 
|  | } | 
|  | /* we only need one vector */ | 
|  | nvec = pci_alloc_irq_vectors(adev->pdev, 1, 1, flags); | 
|  | if (nvec > 0) { | 
|  | adev->irq.msi_enabled = true; | 
|  | dev_dbg(adev->dev, "using MSI/MSI-X.\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!amdgpu_device_has_dc_support(adev)) { | 
|  | if (!adev->enable_virtual_display) | 
|  | /* Disable vblank IRQs aggressively for power-saving */ | 
|  | /* XXX: can this be enabled for DC? */ | 
|  | adev_to_drm(adev)->vblank_disable_immediate = true; | 
|  |  | 
|  | r = drm_vblank_init(adev_to_drm(adev), adev->mode_info.num_crtc); | 
|  | if (r) | 
|  | return r; | 
|  |  | 
|  | /* Pre-DCE11 */ | 
|  | INIT_WORK(&adev->hotplug_work, | 
|  | amdgpu_hotplug_work_func); | 
|  | } | 
|  |  | 
|  | INIT_WORK(&adev->irq.ih1_work, amdgpu_irq_handle_ih1); | 
|  | INIT_WORK(&adev->irq.ih2_work, amdgpu_irq_handle_ih2); | 
|  | INIT_WORK(&adev->irq.ih_soft_work, amdgpu_irq_handle_ih_soft); | 
|  |  | 
|  | /* Use vector 0 for MSI-X. */ | 
|  | r = pci_irq_vector(adev->pdev, 0); | 
|  | if (r < 0) | 
|  | return r; | 
|  | irq = r; | 
|  |  | 
|  | /* PCI devices require shared interrupts. */ | 
|  | r = request_irq(irq, amdgpu_irq_handler, IRQF_SHARED, adev_to_drm(adev)->driver->name, | 
|  | adev_to_drm(adev)); | 
|  | if (r) { | 
|  | if (!amdgpu_device_has_dc_support(adev)) | 
|  | flush_work(&adev->hotplug_work); | 
|  | return r; | 
|  | } | 
|  | adev->irq.installed = true; | 
|  | adev->irq.irq = irq; | 
|  | adev_to_drm(adev)->max_vblank_count = 0x00ffffff; | 
|  |  | 
|  | DRM_DEBUG("amdgpu: irq initialized.\n"); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  |  | 
|  | void amdgpu_irq_fini_hw(struct amdgpu_device *adev) | 
|  | { | 
|  | if (adev->irq.installed) { | 
|  | free_irq(adev->irq.irq, adev_to_drm(adev)); | 
|  | adev->irq.installed = false; | 
|  | if (adev->irq.msi_enabled) | 
|  | pci_free_irq_vectors(adev->pdev); | 
|  |  | 
|  | if (!amdgpu_device_has_dc_support(adev)) | 
|  | flush_work(&adev->hotplug_work); | 
|  | } | 
|  |  | 
|  | amdgpu_ih_ring_fini(adev, &adev->irq.ih_soft); | 
|  | amdgpu_ih_ring_fini(adev, &adev->irq.ih); | 
|  | amdgpu_ih_ring_fini(adev, &adev->irq.ih1); | 
|  | amdgpu_ih_ring_fini(adev, &adev->irq.ih2); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_fini - shut down interrupt handling | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * | 
|  | * Tears down work functions for hotplug and reset interrupts, disables MSI | 
|  | * functionality, shuts down vblank, hotplug and reset interrupt handling, | 
|  | * turns off interrupts from all sources (all ASICs). | 
|  | */ | 
|  | void amdgpu_irq_fini_sw(struct amdgpu_device *adev) | 
|  | { | 
|  | unsigned i, j; | 
|  |  | 
|  | for (i = 0; i < AMDGPU_IRQ_CLIENTID_MAX; ++i) { | 
|  | if (!adev->irq.client[i].sources) | 
|  | continue; | 
|  |  | 
|  | for (j = 0; j < AMDGPU_MAX_IRQ_SRC_ID; ++j) { | 
|  | struct amdgpu_irq_src *src = adev->irq.client[i].sources[j]; | 
|  |  | 
|  | if (!src) | 
|  | continue; | 
|  |  | 
|  | kfree(src->enabled_types); | 
|  | src->enabled_types = NULL; | 
|  | } | 
|  | kfree(adev->irq.client[i].sources); | 
|  | adev->irq.client[i].sources = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_add_id - register IRQ source | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * @client_id: client id | 
|  | * @src_id: source id | 
|  | * @source: IRQ source pointer | 
|  | * | 
|  | * Registers IRQ source on a client. | 
|  | * | 
|  | * Returns: | 
|  | * 0 on success or error code otherwise | 
|  | */ | 
|  | int amdgpu_irq_add_id(struct amdgpu_device *adev, | 
|  | unsigned client_id, unsigned src_id, | 
|  | struct amdgpu_irq_src *source) | 
|  | { | 
|  | if (client_id >= AMDGPU_IRQ_CLIENTID_MAX) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (src_id >= AMDGPU_MAX_IRQ_SRC_ID) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (!source->funcs) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (!adev->irq.client[client_id].sources) { | 
|  | adev->irq.client[client_id].sources = | 
|  | kcalloc(AMDGPU_MAX_IRQ_SRC_ID, | 
|  | sizeof(struct amdgpu_irq_src *), | 
|  | GFP_KERNEL); | 
|  | if (!adev->irq.client[client_id].sources) | 
|  | return -ENOMEM; | 
|  | } | 
|  |  | 
|  | if (adev->irq.client[client_id].sources[src_id] != NULL) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (source->num_types && !source->enabled_types) { | 
|  | atomic_t *types; | 
|  |  | 
|  | types = kcalloc(source->num_types, sizeof(atomic_t), | 
|  | GFP_KERNEL); | 
|  | if (!types) | 
|  | return -ENOMEM; | 
|  |  | 
|  | source->enabled_types = types; | 
|  | } | 
|  |  | 
|  | adev->irq.client[client_id].sources[src_id] = source; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_dispatch - dispatch IRQ to IP blocks | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * @ih: interrupt ring instance | 
|  | * | 
|  | * Dispatches IRQ to IP blocks. | 
|  | */ | 
|  | void amdgpu_irq_dispatch(struct amdgpu_device *adev, | 
|  | struct amdgpu_ih_ring *ih) | 
|  | { | 
|  | u32 ring_index = ih->rptr >> 2; | 
|  | struct amdgpu_iv_entry entry; | 
|  | unsigned client_id, src_id; | 
|  | struct amdgpu_irq_src *src; | 
|  | bool handled = false; | 
|  | int r; | 
|  |  | 
|  | entry.ih = ih; | 
|  | entry.iv_entry = (const uint32_t *)&ih->ring[ring_index]; | 
|  |  | 
|  | /* | 
|  | * timestamp is not supported on some legacy SOCs (cik, cz, iceland, | 
|  | * si and tonga), so initialize timestamp and timestamp_src to 0 | 
|  | */ | 
|  | entry.timestamp = 0; | 
|  | entry.timestamp_src = 0; | 
|  |  | 
|  | amdgpu_ih_decode_iv(adev, &entry); | 
|  |  | 
|  | trace_amdgpu_iv(ih - &adev->irq.ih, &entry); | 
|  |  | 
|  | client_id = entry.client_id; | 
|  | src_id = entry.src_id; | 
|  |  | 
|  | if (client_id >= AMDGPU_IRQ_CLIENTID_MAX) { | 
|  | DRM_DEBUG("Invalid client_id in IV: %d\n", client_id); | 
|  |  | 
|  | } else	if (src_id >= AMDGPU_MAX_IRQ_SRC_ID) { | 
|  | DRM_DEBUG("Invalid src_id in IV: %d\n", src_id); | 
|  |  | 
|  | } else if ((client_id == AMDGPU_IRQ_CLIENTID_LEGACY) && | 
|  | adev->irq.virq[src_id]) { | 
|  | generic_handle_domain_irq(adev->irq.domain, src_id); | 
|  |  | 
|  | } else if (!adev->irq.client[client_id].sources) { | 
|  | DRM_DEBUG("Unregistered interrupt client_id: %d src_id: %d\n", | 
|  | client_id, src_id); | 
|  |  | 
|  | } else if ((src = adev->irq.client[client_id].sources[src_id])) { | 
|  | r = src->funcs->process(adev, src, &entry); | 
|  | if (r < 0) | 
|  | DRM_ERROR("error processing interrupt (%d)\n", r); | 
|  | else if (r) | 
|  | handled = true; | 
|  |  | 
|  | } else { | 
|  | DRM_DEBUG("Unhandled interrupt src_id: %d\n", src_id); | 
|  | } | 
|  |  | 
|  | /* Send it to amdkfd as well if it isn't already handled */ | 
|  | if (!handled) | 
|  | amdgpu_amdkfd_interrupt(adev, entry.iv_entry); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_delegate - delegate IV to soft IH ring | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * @entry: IV entry | 
|  | * @num_dw: size of IV | 
|  | * | 
|  | * Delegate the IV to the soft IH ring and schedule processing of it. Used | 
|  | * if the hardware delegation to IH1 or IH2 doesn't work for some reason. | 
|  | */ | 
|  | void amdgpu_irq_delegate(struct amdgpu_device *adev, | 
|  | struct amdgpu_iv_entry *entry, | 
|  | unsigned int num_dw) | 
|  | { | 
|  | amdgpu_ih_ring_write(&adev->irq.ih_soft, entry->iv_entry, num_dw); | 
|  | schedule_work(&adev->irq.ih_soft_work); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_update - update hardware interrupt state | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * @src: interrupt source pointer | 
|  | * @type: type of interrupt | 
|  | * | 
|  | * Updates interrupt state for the specific source (all ASICs). | 
|  | */ | 
|  | int amdgpu_irq_update(struct amdgpu_device *adev, | 
|  | struct amdgpu_irq_src *src, unsigned type) | 
|  | { | 
|  | unsigned long irqflags; | 
|  | enum amdgpu_interrupt_state state; | 
|  | int r; | 
|  |  | 
|  | spin_lock_irqsave(&adev->irq.lock, irqflags); | 
|  |  | 
|  | /* We need to determine after taking the lock, otherwise | 
|  | we might disable just enabled interrupts again */ | 
|  | if (amdgpu_irq_enabled(adev, src, type)) | 
|  | state = AMDGPU_IRQ_STATE_ENABLE; | 
|  | else | 
|  | state = AMDGPU_IRQ_STATE_DISABLE; | 
|  |  | 
|  | r = src->funcs->set(adev, src, type, state); | 
|  | spin_unlock_irqrestore(&adev->irq.lock, irqflags); | 
|  | return r; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_gpu_reset_resume_helper - update interrupt states on all sources | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * | 
|  | * Updates state of all types of interrupts on all sources on resume after | 
|  | * reset. | 
|  | */ | 
|  | void amdgpu_irq_gpu_reset_resume_helper(struct amdgpu_device *adev) | 
|  | { | 
|  | int i, j, k; | 
|  |  | 
|  | if (amdgpu_sriov_vf(adev) || amdgpu_passthrough(adev)) | 
|  | amdgpu_restore_msix(adev); | 
|  |  | 
|  | for (i = 0; i < AMDGPU_IRQ_CLIENTID_MAX; ++i) { | 
|  | if (!adev->irq.client[i].sources) | 
|  | continue; | 
|  |  | 
|  | for (j = 0; j < AMDGPU_MAX_IRQ_SRC_ID; ++j) { | 
|  | struct amdgpu_irq_src *src = adev->irq.client[i].sources[j]; | 
|  |  | 
|  | if (!src || !src->funcs || !src->funcs->set) | 
|  | continue; | 
|  | for (k = 0; k < src->num_types; k++) | 
|  | amdgpu_irq_update(adev, src, k); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_get - enable interrupt | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * @src: interrupt source pointer | 
|  | * @type: type of interrupt | 
|  | * | 
|  | * Enables specified type of interrupt on the specified source (all ASICs). | 
|  | * | 
|  | * Returns: | 
|  | * 0 on success or error code otherwise | 
|  | */ | 
|  | int amdgpu_irq_get(struct amdgpu_device *adev, struct amdgpu_irq_src *src, | 
|  | unsigned type) | 
|  | { | 
|  | if (!adev->irq.installed) | 
|  | return -ENOENT; | 
|  |  | 
|  | if (type >= src->num_types) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (!src->enabled_types || !src->funcs->set) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (atomic_inc_return(&src->enabled_types[type]) == 1) | 
|  | return amdgpu_irq_update(adev, src, type); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_put - disable interrupt | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * @src: interrupt source pointer | 
|  | * @type: type of interrupt | 
|  | * | 
|  | * Enables specified type of interrupt on the specified source (all ASICs). | 
|  | * | 
|  | * Returns: | 
|  | * 0 on success or error code otherwise | 
|  | */ | 
|  | int amdgpu_irq_put(struct amdgpu_device *adev, struct amdgpu_irq_src *src, | 
|  | unsigned type) | 
|  | { | 
|  | if (!adev->irq.installed) | 
|  | return -ENOENT; | 
|  |  | 
|  | if (type >= src->num_types) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (!src->enabled_types || !src->funcs->set) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (atomic_dec_and_test(&src->enabled_types[type])) | 
|  | return amdgpu_irq_update(adev, src, type); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_enabled - check whether interrupt is enabled or not | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * @src: interrupt source pointer | 
|  | * @type: type of interrupt | 
|  | * | 
|  | * Checks whether the given type of interrupt is enabled on the given source. | 
|  | * | 
|  | * Returns: | 
|  | * *true* if interrupt is enabled, *false* if interrupt is disabled or on | 
|  | * invalid parameters | 
|  | */ | 
|  | bool amdgpu_irq_enabled(struct amdgpu_device *adev, struct amdgpu_irq_src *src, | 
|  | unsigned type) | 
|  | { | 
|  | if (!adev->irq.installed) | 
|  | return false; | 
|  |  | 
|  | if (type >= src->num_types) | 
|  | return false; | 
|  |  | 
|  | if (!src->enabled_types || !src->funcs->set) | 
|  | return false; | 
|  |  | 
|  | return !!atomic_read(&src->enabled_types[type]); | 
|  | } | 
|  |  | 
|  | /* XXX: Generic IRQ handling */ | 
|  | static void amdgpu_irq_mask(struct irq_data *irqd) | 
|  | { | 
|  | /* XXX */ | 
|  | } | 
|  |  | 
|  | static void amdgpu_irq_unmask(struct irq_data *irqd) | 
|  | { | 
|  | /* XXX */ | 
|  | } | 
|  |  | 
|  | /* amdgpu hardware interrupt chip descriptor */ | 
|  | static struct irq_chip amdgpu_irq_chip = { | 
|  | .name = "amdgpu-ih", | 
|  | .irq_mask = amdgpu_irq_mask, | 
|  | .irq_unmask = amdgpu_irq_unmask, | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * amdgpu_irqdomain_map - create mapping between virtual and hardware IRQ numbers | 
|  | * | 
|  | * @d: amdgpu IRQ domain pointer (unused) | 
|  | * @irq: virtual IRQ number | 
|  | * @hwirq: hardware irq number | 
|  | * | 
|  | * Current implementation assigns simple interrupt handler to the given virtual | 
|  | * IRQ. | 
|  | * | 
|  | * Returns: | 
|  | * 0 on success or error code otherwise | 
|  | */ | 
|  | static int amdgpu_irqdomain_map(struct irq_domain *d, | 
|  | unsigned int irq, irq_hw_number_t hwirq) | 
|  | { | 
|  | if (hwirq >= AMDGPU_MAX_IRQ_SRC_ID) | 
|  | return -EPERM; | 
|  |  | 
|  | irq_set_chip_and_handler(irq, | 
|  | &amdgpu_irq_chip, handle_simple_irq); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Implementation of methods for amdgpu IRQ domain */ | 
|  | static const struct irq_domain_ops amdgpu_hw_irqdomain_ops = { | 
|  | .map = amdgpu_irqdomain_map, | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_add_domain - create a linear IRQ domain | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * | 
|  | * Creates an IRQ domain for GPU interrupt sources | 
|  | * that may be driven by another driver (e.g., ACP). | 
|  | * | 
|  | * Returns: | 
|  | * 0 on success or error code otherwise | 
|  | */ | 
|  | int amdgpu_irq_add_domain(struct amdgpu_device *adev) | 
|  | { | 
|  | adev->irq.domain = irq_domain_add_linear(NULL, AMDGPU_MAX_IRQ_SRC_ID, | 
|  | &amdgpu_hw_irqdomain_ops, adev); | 
|  | if (!adev->irq.domain) { | 
|  | DRM_ERROR("GPU irq add domain failed\n"); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_remove_domain - remove the IRQ domain | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * | 
|  | * Removes the IRQ domain for GPU interrupt sources | 
|  | * that may be driven by another driver (e.g., ACP). | 
|  | */ | 
|  | void amdgpu_irq_remove_domain(struct amdgpu_device *adev) | 
|  | { | 
|  | if (adev->irq.domain) { | 
|  | irq_domain_remove(adev->irq.domain); | 
|  | adev->irq.domain = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * amdgpu_irq_create_mapping - create mapping between domain Linux IRQs | 
|  | * | 
|  | * @adev: amdgpu device pointer | 
|  | * @src_id: IH source id | 
|  | * | 
|  | * Creates mapping between a domain IRQ (GPU IH src id) and a Linux IRQ | 
|  | * Use this for components that generate a GPU interrupt, but are driven | 
|  | * by a different driver (e.g., ACP). | 
|  | * | 
|  | * Returns: | 
|  | * Linux IRQ | 
|  | */ | 
|  | unsigned amdgpu_irq_create_mapping(struct amdgpu_device *adev, unsigned src_id) | 
|  | { | 
|  | adev->irq.virq[src_id] = irq_create_mapping(adev->irq.domain, src_id); | 
|  |  | 
|  | return adev->irq.virq[src_id]; | 
|  | } |