/* * Copyright (c) 2019, NVIDIA CORPORATION. All rights reserved. * * 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 AUTHORS OR COPYRIGHT HOLDERS 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. */ #include #include #include #include #include #include #include static int gr_intr_handle_tpc_exception(struct gk20a *g, u32 gpc, u32 tpc, bool *post_event, struct channel_gk20a *fault_ch, u32 *hww_global_esr) { int tmp_ret, ret = 0; struct nvgpu_gr_tpc_exception pending_tpc; u32 offset = nvgpu_gr_gpc_offset(g, gpc) + nvgpu_gr_tpc_offset(g, tpc); u32 tpc_exception = g->ops.gr.intr.get_tpc_exception(g, offset, &pending_tpc); u32 sm_per_tpc = nvgpu_get_litter_value(g, GPU_LIT_NUM_SM_PER_TPC); nvgpu_log(g, gpu_dbg_intr | gpu_dbg_gpu_dbg, "GPC%d TPC%d: pending exception 0x%x", gpc, tpc, tpc_exception); /* check if an sm exeption is pending */ if (pending_tpc.sm_exception) { u32 esr_sm_sel, sm; nvgpu_log(g, gpu_dbg_intr | gpu_dbg_gpu_dbg, "GPC%d TPC%d: SM exception pending", gpc, tpc); if (g->ops.gr.handle_tpc_sm_ecc_exception != NULL) { g->ops.gr.handle_tpc_sm_ecc_exception(g, gpc, tpc, post_event, fault_ch, hww_global_esr); } g->ops.gr.get_esr_sm_sel(g, gpc, tpc, &esr_sm_sel); for (sm = 0; sm < sm_per_tpc; sm++) { if ((esr_sm_sel & BIT32(sm)) == 0U) { continue; } nvgpu_log(g, gpu_dbg_intr | gpu_dbg_gpu_dbg, "GPC%d TPC%d: SM%d exception pending", gpc, tpc, sm); tmp_ret = g->ops.gr.handle_sm_exception(g, gpc, tpc, sm, post_event, fault_ch, hww_global_esr); ret = (ret != 0) ? ret : tmp_ret; /* clear the hwws, also causes tpc and gpc * exceptions to be cleared. Should be cleared * only if SM is locked down or empty. */ g->ops.gr.clear_sm_hww(g, gpc, tpc, sm, *hww_global_esr); } } /* check if a tex exception is pending */ if (pending_tpc.tex_exception) { nvgpu_log(g, gpu_dbg_intr | gpu_dbg_gpu_dbg, "GPC%d TPC%d: TEX exception pending", gpc, tpc); if (g->ops.gr.intr.handle_tex_exception != NULL) { g->ops.gr.intr.handle_tex_exception(g, gpc, tpc); } } /* check if a mpc exception is pending */ if (pending_tpc.mpc_exception) { nvgpu_log(g, gpu_dbg_intr | gpu_dbg_gpu_dbg, "GPC%d TPC%d: MPC exception pending", gpc, tpc); if (g->ops.gr.intr.handle_tpc_mpc_exception != NULL) { g->ops.gr.intr.handle_tpc_mpc_exception(g, gpc, tpc); } } return ret; } int nvgpu_gr_intr_handle_gpc_exception(struct gk20a *g, bool *post_event, struct nvgpu_gr_config *gr_config, struct channel_gk20a *fault_ch, u32 *hww_global_esr) { int tmp_ret, ret = 0; u32 gpc, tpc; u32 exception1 = g->ops.gr.intr.read_exception1(g); u32 gpc_exception, tpc_exception; nvgpu_log(g, gpu_dbg_intr | gpu_dbg_gpu_dbg, " "); for (gpc = 0; gpc < nvgpu_gr_config_get_gpc_count(gr_config); gpc++) { if ((exception1 & BIT32(gpc)) == 0U) { continue; } nvgpu_log(g, gpu_dbg_intr | gpu_dbg_gpu_dbg, "GPC%d exception pending", gpc); gpc_exception = g->ops.gr.intr.read_gpc_exception(g, gpc); tpc_exception = g->ops.gr.intr.read_gpc_tpc_exception( gpc_exception); /* check if any tpc has an exception */ for (tpc = 0; tpc < nvgpu_gr_config_get_gpc_tpc_count(gr_config, gpc); tpc++) { if ((tpc_exception & BIT32(tpc)) == 0U) { continue; } nvgpu_log(g, gpu_dbg_intr | gpu_dbg_gpu_dbg, "GPC%d: TPC%d exception pending", gpc, tpc); tmp_ret = gr_intr_handle_tpc_exception(g, gpc, tpc, post_event, fault_ch, hww_global_esr); ret = (ret != 0) ? ret : tmp_ret; } /* Handle GCC exception */ if (g->ops.gr.intr.handle_gcc_exception != NULL) { g->ops.gr.intr.handle_gcc_exception(g, gpc, tpc, gpc_exception, &g->ecc.gr.gcc_l15_ecc_corrected_err_count[gpc].counter, &g->ecc.gr.gcc_l15_ecc_uncorrected_err_count[gpc].counter); } /* Handle GPCCS exceptions */ if (g->ops.gr.intr.handle_gpc_gpccs_exception != NULL) { g->ops.gr.intr.handle_gpc_gpccs_exception(g, gpc, gpc_exception, &g->ecc.gr.gpccs_ecc_corrected_err_count[gpc].counter, &g->ecc.gr.gpccs_ecc_uncorrected_err_count[gpc].counter); } /* Handle GPCMMU exceptions */ if (g->ops.gr.intr.handle_gpc_gpcmmu_exception != NULL) { g->ops.gr.intr.handle_gpc_gpcmmu_exception(g, gpc, gpc_exception, &g->ecc.gr.mmu_l1tlb_ecc_corrected_err_count[gpc].counter, &g->ecc.gr.mmu_l1tlb_ecc_uncorrected_err_count[gpc].counter); } } return ret; } #if defined(CONFIG_GK20A_CYCLE_STATS) static inline bool is_valid_cyclestats_bar0_offset_gk20a(struct gk20a *g, u32 offset) { /* support only 24-bit 4-byte aligned offsets */ bool valid = !(offset & 0xFF000003U); if (g->allow_all) { return true; } /* whitelist check */ valid = valid && is_bar0_global_offset_whitelisted_gk20a(g, offset); /* resource size check in case there was a problem * with allocating the assumed size of bar0 */ valid = valid && nvgpu_io_valid_reg(g, offset); return valid; } #endif int nvgpu_gr_intr_handle_notify_pending(struct gk20a *g, struct nvgpu_gr_isr_data *isr_data) { struct channel_gk20a *ch = isr_data->ch; #if defined(CONFIG_GK20A_CYCLE_STATS) void *virtual_address; u32 buffer_size; u32 offset; bool exit; #endif if (ch == NULL || tsg_gk20a_from_ch(ch) == NULL) { return 0; } #if defined(CONFIG_GK20A_CYCLE_STATS) /* GL will never use payload 0 for cycle state */ if ((ch->cyclestate.cyclestate_buffer == NULL) || (isr_data->data_lo == 0)) { return 0; } nvgpu_mutex_acquire(&ch->cyclestate.cyclestate_buffer_mutex); virtual_address = ch->cyclestate.cyclestate_buffer; buffer_size = ch->cyclestate.cyclestate_buffer_size; offset = isr_data->data_lo; exit = false; while (!exit) { struct share_buffer_head *sh_hdr; u32 min_element_size; /* validate offset */ if (offset + sizeof(struct share_buffer_head) > buffer_size || offset + sizeof(struct share_buffer_head) < offset) { nvgpu_err(g, "cyclestats buffer overrun at offset 0x%x", offset); break; } sh_hdr = (struct share_buffer_head *) ((char *)virtual_address + offset); min_element_size = (sh_hdr->operation == OP_END ? sizeof(struct share_buffer_head) : sizeof(struct gk20a_cyclestate_buffer_elem)); /* validate sh_hdr->size */ if (sh_hdr->size < min_element_size || offset + sh_hdr->size > buffer_size || offset + sh_hdr->size < offset) { nvgpu_err(g, "bad cyclestate buffer header size at offset 0x%x", offset); sh_hdr->failed = true; break; } switch (sh_hdr->operation) { case OP_END: exit = true; break; case BAR0_READ32: case BAR0_WRITE32: { struct gk20a_cyclestate_buffer_elem *op_elem = (struct gk20a_cyclestate_buffer_elem *)sh_hdr; bool valid = is_valid_cyclestats_bar0_offset_gk20a( g, op_elem->offset_bar0); u32 raw_reg; u64 mask_orig; u64 v; if (!valid) { nvgpu_err(g, "invalid cycletstats op offset: 0x%x", op_elem->offset_bar0); sh_hdr->failed = exit = true; break; } mask_orig = ((1ULL << (op_elem->last_bit + 1)) - 1) & ~((1ULL << op_elem->first_bit) - 1); raw_reg = nvgpu_readl(g, op_elem->offset_bar0); switch (sh_hdr->operation) { case BAR0_READ32: op_elem->data = ((raw_reg & mask_orig) >> op_elem->first_bit); break; case BAR0_WRITE32: v = 0; if ((unsigned int)mask_orig != ~((unsigned int)0)) { v = (unsigned int) (raw_reg & ~mask_orig); } v |= ((op_elem->data << op_elem->first_bit) & mask_orig); nvgpu_writel(g,op_elem->offset_bar0, (unsigned int)v); break; default: /* nop ok?*/ break; } } break; default: /* no operation content case */ exit = true; break; } sh_hdr->completed = true; offset += sh_hdr->size; } nvgpu_mutex_release(&ch->cyclestate.cyclestate_buffer_mutex); #endif nvgpu_log_fn(g, " "); nvgpu_cond_broadcast_interruptible(&ch->notifier_wq); return 0; } int nvgpu_gr_intr_handle_semaphore_pending(struct gk20a *g, struct nvgpu_gr_isr_data *isr_data) { struct channel_gk20a *ch = isr_data->ch; struct tsg_gk20a *tsg; if (ch == NULL) { return 0; } tsg = tsg_gk20a_from_ch(ch); if (tsg != NULL) { g->ops.tsg.post_event_id(tsg, NVGPU_EVENT_ID_GR_SEMAPHORE_WRITE_AWAKEN); nvgpu_cond_broadcast(&ch->semaphore_wq); } else { nvgpu_err(g, "chid: %d is not bound to tsg", ch->chid); } return 0; }