1 /* 2 * Copyright 2020 Advanced Micro Devices, Inc. 3 * 4 * Permission is hereby granted, free of charge, to any person obtaining a 5 * copy of this software and associated documentation files (the "Software"), 6 * to deal in the Software without restriction, including without limitation 7 * the rights to use, copy, modify, merge, publish, distribute, sublicense, 8 * and/or sell copies of the Software, and to permit persons to whom the 9 * Software is furnished to do so, subject to the following conditions: 10 * 11 * The above copyright notice and this permission notice shall be included in 12 * all copies or substantial portions of the Software. 13 * 14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR 18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 20 * OTHER DEALINGS IN THE SOFTWARE. 21 * 22 * Authors: AMD 23 * 24 */ 25 26 27 #include "dc_bios_types.h" 28 #include "dcn30_vpg.h" 29 #include "reg_helper.h" 30 31 #define DC_LOGGER \ 32 vpg3->base.ctx->logger 33 34 #define REG(reg)\ 35 (vpg3->regs->reg) 36 37 #undef FN 38 #define FN(reg_name, field_name) \ 39 vpg3->vpg_shift->field_name, vpg3->vpg_mask->field_name 40 41 42 #define CTX \ 43 vpg3->base.ctx 44 45 46 static void vpg3_update_generic_info_packet( 47 struct vpg *vpg, 48 uint32_t packet_index, 49 const struct dc_info_packet *info_packet) 50 { 51 struct dcn30_vpg *vpg3 = DCN30_VPG_FROM_VPG(vpg); 52 uint32_t i; 53 54 /* TODOFPGA Figure out a proper number for max_retries polling for lock 55 * use 50 for now. 56 */ 57 uint32_t max_retries = 50; 58 59 if (packet_index > 14) 60 ASSERT(0); 61 62 /* poll dig_update_lock is not locked -> asic internal signal 63 * assume otg master lock will unlock it 64 */ 65 /* REG_WAIT(AFMT_VBI_PACKET_CONTROL, AFMT_GENERIC_LOCK_STATUS, 66 * 0, 10, max_retries); 67 */ 68 69 /* TODO: Check if this is required */ 70 /* check if HW reading GSP memory */ 71 REG_WAIT(VPG_GENERIC_STATUS, VPG_GENERIC_CONFLICT_OCCURED, 72 0, 10, max_retries); 73 74 /* HW does is not reading GSP memory not reading too long -> 75 * something wrong. clear GPS memory access and notify? 76 * hw SW is writing to GSP memory 77 */ 78 REG_UPDATE(VPG_GENERIC_STATUS, VPG_GENERIC_CONFLICT_CLR, 1); 79 80 /* choose which generic packet to use */ 81 REG_UPDATE(VPG_GENERIC_PACKET_ACCESS_CTRL, 82 VPG_GENERIC_DATA_INDEX, packet_index*9); 83 84 /* write generic packet header 85 * (4th byte is for GENERIC0 only) 86 */ 87 REG_SET_4(VPG_GENERIC_PACKET_DATA, 0, 88 VPG_GENERIC_DATA_BYTE0, info_packet->hb0, 89 VPG_GENERIC_DATA_BYTE1, info_packet->hb1, 90 VPG_GENERIC_DATA_BYTE2, info_packet->hb2, 91 VPG_GENERIC_DATA_BYTE3, info_packet->hb3); 92 93 /* write generic packet contents 94 * (we never use last 4 bytes) 95 * there are 8 (0-7) mmDIG0_AFMT_GENERIC0_x registers 96 */ 97 { 98 const uint32_t *content = 99 (const uint32_t *) &info_packet->sb[0]; 100 101 for (i = 0; i < 8; i++) { 102 REG_WRITE(VPG_GENERIC_PACKET_DATA, *content++); 103 } 104 } 105 106 /* atomically update double-buffered GENERIC0 registers in frame mode 107 * (update at next block_update when block_update_lock == 0). 108 */ 109 switch (packet_index) { 110 case 0: 111 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 112 VPG_GENERIC0_FRAME_UPDATE, 1); 113 break; 114 case 1: 115 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 116 VPG_GENERIC1_FRAME_UPDATE, 1); 117 break; 118 case 2: 119 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 120 VPG_GENERIC2_FRAME_UPDATE, 1); 121 break; 122 case 3: 123 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 124 VPG_GENERIC3_FRAME_UPDATE, 1); 125 break; 126 case 4: 127 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 128 VPG_GENERIC4_FRAME_UPDATE, 1); 129 break; 130 case 5: 131 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 132 VPG_GENERIC5_FRAME_UPDATE, 1); 133 break; 134 case 6: 135 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 136 VPG_GENERIC6_FRAME_UPDATE, 1); 137 break; 138 case 7: 139 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 140 VPG_GENERIC7_FRAME_UPDATE, 1); 141 break; 142 case 8: 143 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 144 VPG_GENERIC8_FRAME_UPDATE, 1); 145 break; 146 case 9: 147 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 148 VPG_GENERIC9_FRAME_UPDATE, 1); 149 break; 150 case 10: 151 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 152 VPG_GENERIC10_FRAME_UPDATE, 1); 153 break; 154 case 11: 155 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 156 VPG_GENERIC11_FRAME_UPDATE, 1); 157 break; 158 case 12: 159 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 160 VPG_GENERIC12_FRAME_UPDATE, 1); 161 break; 162 case 13: 163 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 164 VPG_GENERIC13_FRAME_UPDATE, 1); 165 break; 166 case 14: 167 REG_UPDATE(VPG_GSP_FRAME_UPDATE_CTRL, 168 VPG_GENERIC14_FRAME_UPDATE, 1); 169 break; 170 default: 171 break; 172 } 173 } 174 175 static struct vpg_funcs dcn30_vpg_funcs = { 176 .update_generic_info_packet = vpg3_update_generic_info_packet, 177 }; 178 179 void vpg3_construct(struct dcn30_vpg *vpg3, 180 struct dc_context *ctx, 181 uint32_t inst, 182 const struct dcn30_vpg_registers *vpg_regs, 183 const struct dcn30_vpg_shift *vpg_shift, 184 const struct dcn30_vpg_mask *vpg_mask) 185 { 186 vpg3->base.ctx = ctx; 187 188 vpg3->base.inst = inst; 189 vpg3->base.funcs = &dcn30_vpg_funcs; 190 191 vpg3->regs = vpg_regs; 192 vpg3->vpg_shift = vpg_shift; 193 vpg3->vpg_mask = vpg_mask; 194 } 195