2 * Copyright 2014 Advanced Micro Devices, Inc.
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:
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
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.
24 #include <linux/firmware.h>
25 #include <linux/slab.h>
26 #include <linux/module.h>
29 #include "amdgpu_ucode.h"
31 static void amdgpu_ucode_print_common_hdr(const struct common_firmware_header
*hdr
)
33 DRM_DEBUG("size_bytes: %u\n", le32_to_cpu(hdr
->size_bytes
));
34 DRM_DEBUG("header_size_bytes: %u\n", le32_to_cpu(hdr
->header_size_bytes
));
35 DRM_DEBUG("header_version_major: %u\n", le16_to_cpu(hdr
->header_version_major
));
36 DRM_DEBUG("header_version_minor: %u\n", le16_to_cpu(hdr
->header_version_minor
));
37 DRM_DEBUG("ip_version_major: %u\n", le16_to_cpu(hdr
->ip_version_major
));
38 DRM_DEBUG("ip_version_minor: %u\n", le16_to_cpu(hdr
->ip_version_minor
));
39 DRM_DEBUG("ucode_version: 0x%08x\n", le32_to_cpu(hdr
->ucode_version
));
40 DRM_DEBUG("ucode_size_bytes: %u\n", le32_to_cpu(hdr
->ucode_size_bytes
));
41 DRM_DEBUG("ucode_array_offset_bytes: %u\n",
42 le32_to_cpu(hdr
->ucode_array_offset_bytes
));
43 DRM_DEBUG("crc32: 0x%08x\n", le32_to_cpu(hdr
->crc32
));
46 void amdgpu_ucode_print_mc_hdr(const struct common_firmware_header
*hdr
)
48 uint16_t version_major
= le16_to_cpu(hdr
->header_version_major
);
49 uint16_t version_minor
= le16_to_cpu(hdr
->header_version_minor
);
52 amdgpu_ucode_print_common_hdr(hdr
);
54 if (version_major
== 1) {
55 const struct mc_firmware_header_v1_0
*mc_hdr
=
56 container_of(hdr
, struct mc_firmware_header_v1_0
, header
);
58 DRM_DEBUG("io_debug_size_bytes: %u\n",
59 le32_to_cpu(mc_hdr
->io_debug_size_bytes
));
60 DRM_DEBUG("io_debug_array_offset_bytes: %u\n",
61 le32_to_cpu(mc_hdr
->io_debug_array_offset_bytes
));
63 DRM_ERROR("Unknown MC ucode version: %u.%u\n", version_major
, version_minor
);
67 void amdgpu_ucode_print_smc_hdr(const struct common_firmware_header
*hdr
)
69 uint16_t version_major
= le16_to_cpu(hdr
->header_version_major
);
70 uint16_t version_minor
= le16_to_cpu(hdr
->header_version_minor
);
73 amdgpu_ucode_print_common_hdr(hdr
);
75 if (version_major
== 1) {
76 const struct smc_firmware_header_v1_0
*smc_hdr
=
77 container_of(hdr
, struct smc_firmware_header_v1_0
, header
);
79 DRM_DEBUG("ucode_start_addr: %u\n", le32_to_cpu(smc_hdr
->ucode_start_addr
));
81 DRM_ERROR("Unknown SMC ucode version: %u.%u\n", version_major
, version_minor
);
85 void amdgpu_ucode_print_gfx_hdr(const struct common_firmware_header
*hdr
)
87 uint16_t version_major
= le16_to_cpu(hdr
->header_version_major
);
88 uint16_t version_minor
= le16_to_cpu(hdr
->header_version_minor
);
91 amdgpu_ucode_print_common_hdr(hdr
);
93 if (version_major
== 1) {
94 const struct gfx_firmware_header_v1_0
*gfx_hdr
=
95 container_of(hdr
, struct gfx_firmware_header_v1_0
, header
);
97 DRM_DEBUG("ucode_feature_version: %u\n",
98 le32_to_cpu(gfx_hdr
->ucode_feature_version
));
99 DRM_DEBUG("jt_offset: %u\n", le32_to_cpu(gfx_hdr
->jt_offset
));
100 DRM_DEBUG("jt_size: %u\n", le32_to_cpu(gfx_hdr
->jt_size
));
102 DRM_ERROR("Unknown GFX ucode version: %u.%u\n", version_major
, version_minor
);
106 void amdgpu_ucode_print_rlc_hdr(const struct common_firmware_header
*hdr
)
108 uint16_t version_major
= le16_to_cpu(hdr
->header_version_major
);
109 uint16_t version_minor
= le16_to_cpu(hdr
->header_version_minor
);
112 amdgpu_ucode_print_common_hdr(hdr
);
114 if (version_major
== 1) {
115 const struct rlc_firmware_header_v1_0
*rlc_hdr
=
116 container_of(hdr
, struct rlc_firmware_header_v1_0
, header
);
118 DRM_DEBUG("ucode_feature_version: %u\n",
119 le32_to_cpu(rlc_hdr
->ucode_feature_version
));
120 DRM_DEBUG("save_and_restore_offset: %u\n",
121 le32_to_cpu(rlc_hdr
->save_and_restore_offset
));
122 DRM_DEBUG("clear_state_descriptor_offset: %u\n",
123 le32_to_cpu(rlc_hdr
->clear_state_descriptor_offset
));
124 DRM_DEBUG("avail_scratch_ram_locations: %u\n",
125 le32_to_cpu(rlc_hdr
->avail_scratch_ram_locations
));
126 DRM_DEBUG("master_pkt_description_offset: %u\n",
127 le32_to_cpu(rlc_hdr
->master_pkt_description_offset
));
128 } else if (version_major
== 2) {
129 const struct rlc_firmware_header_v2_0
*rlc_hdr
=
130 container_of(hdr
, struct rlc_firmware_header_v2_0
, header
);
132 DRM_DEBUG("ucode_feature_version: %u\n",
133 le32_to_cpu(rlc_hdr
->ucode_feature_version
));
134 DRM_DEBUG("jt_offset: %u\n", le32_to_cpu(rlc_hdr
->jt_offset
));
135 DRM_DEBUG("jt_size: %u\n", le32_to_cpu(rlc_hdr
->jt_size
));
136 DRM_DEBUG("save_and_restore_offset: %u\n",
137 le32_to_cpu(rlc_hdr
->save_and_restore_offset
));
138 DRM_DEBUG("clear_state_descriptor_offset: %u\n",
139 le32_to_cpu(rlc_hdr
->clear_state_descriptor_offset
));
140 DRM_DEBUG("avail_scratch_ram_locations: %u\n",
141 le32_to_cpu(rlc_hdr
->avail_scratch_ram_locations
));
142 DRM_DEBUG("reg_restore_list_size: %u\n",
143 le32_to_cpu(rlc_hdr
->reg_restore_list_size
));
144 DRM_DEBUG("reg_list_format_start: %u\n",
145 le32_to_cpu(rlc_hdr
->reg_list_format_start
));
146 DRM_DEBUG("reg_list_format_separate_start: %u\n",
147 le32_to_cpu(rlc_hdr
->reg_list_format_separate_start
));
148 DRM_DEBUG("starting_offsets_start: %u\n",
149 le32_to_cpu(rlc_hdr
->starting_offsets_start
));
150 DRM_DEBUG("reg_list_format_size_bytes: %u\n",
151 le32_to_cpu(rlc_hdr
->reg_list_format_size_bytes
));
152 DRM_DEBUG("reg_list_format_array_offset_bytes: %u\n",
153 le32_to_cpu(rlc_hdr
->reg_list_format_array_offset_bytes
));
154 DRM_DEBUG("reg_list_size_bytes: %u\n",
155 le32_to_cpu(rlc_hdr
->reg_list_size_bytes
));
156 DRM_DEBUG("reg_list_array_offset_bytes: %u\n",
157 le32_to_cpu(rlc_hdr
->reg_list_array_offset_bytes
));
158 DRM_DEBUG("reg_list_format_separate_size_bytes: %u\n",
159 le32_to_cpu(rlc_hdr
->reg_list_format_separate_size_bytes
));
160 DRM_DEBUG("reg_list_format_separate_array_offset_bytes: %u\n",
161 le32_to_cpu(rlc_hdr
->reg_list_format_separate_array_offset_bytes
));
162 DRM_DEBUG("reg_list_separate_size_bytes: %u\n",
163 le32_to_cpu(rlc_hdr
->reg_list_separate_size_bytes
));
164 DRM_DEBUG("reg_list_separate_size_bytes: %u\n",
165 le32_to_cpu(rlc_hdr
->reg_list_separate_size_bytes
));
167 DRM_ERROR("Unknown RLC ucode version: %u.%u\n", version_major
, version_minor
);
171 void amdgpu_ucode_print_sdma_hdr(const struct common_firmware_header
*hdr
)
173 uint16_t version_major
= le16_to_cpu(hdr
->header_version_major
);
174 uint16_t version_minor
= le16_to_cpu(hdr
->header_version_minor
);
177 amdgpu_ucode_print_common_hdr(hdr
);
179 if (version_major
== 1) {
180 const struct sdma_firmware_header_v1_0
*sdma_hdr
=
181 container_of(hdr
, struct sdma_firmware_header_v1_0
, header
);
183 DRM_DEBUG("ucode_feature_version: %u\n",
184 le32_to_cpu(sdma_hdr
->ucode_feature_version
));
185 DRM_DEBUG("ucode_change_version: %u\n",
186 le32_to_cpu(sdma_hdr
->ucode_change_version
));
187 DRM_DEBUG("jt_offset: %u\n", le32_to_cpu(sdma_hdr
->jt_offset
));
188 DRM_DEBUG("jt_size: %u\n", le32_to_cpu(sdma_hdr
->jt_size
));
189 if (version_minor
>= 1) {
190 const struct sdma_firmware_header_v1_1
*sdma_v1_1_hdr
=
191 container_of(sdma_hdr
, struct sdma_firmware_header_v1_1
, v1_0
);
192 DRM_DEBUG("digest_size: %u\n", le32_to_cpu(sdma_v1_1_hdr
->digest_size
));
195 DRM_ERROR("Unknown SDMA ucode version: %u.%u\n",
196 version_major
, version_minor
);
200 int amdgpu_ucode_validate(const struct firmware
*fw
)
202 const struct common_firmware_header
*hdr
=
203 (const struct common_firmware_header
*)fw
->data
;
205 if (fw
->size
== le32_to_cpu(hdr
->size_bytes
))
211 bool amdgpu_ucode_hdr_version(union amdgpu_firmware_header
*hdr
,
212 uint16_t hdr_major
, uint16_t hdr_minor
)
214 if ((hdr
->common
.header_version_major
== hdr_major
) &&
215 (hdr
->common
.header_version_minor
== hdr_minor
))
220 static int amdgpu_ucode_init_single_fw(struct amdgpu_firmware_info
*ucode
,
221 uint64_t mc_addr
, void *kptr
)
223 const struct common_firmware_header
*header
= NULL
;
225 if (NULL
== ucode
->fw
)
228 ucode
->mc_addr
= mc_addr
;
231 header
= (const struct common_firmware_header
*)ucode
->fw
->data
;
232 memcpy(ucode
->kaddr
, (void *)((uint8_t *)ucode
->fw
->data
+
233 le32_to_cpu(header
->ucode_array_offset_bytes
)),
234 le32_to_cpu(header
->ucode_size_bytes
));
239 int amdgpu_ucode_init_bo(struct amdgpu_device
*adev
)
241 struct amdgpu_bo
**bo
= &adev
->firmware
.fw_buf
;
243 void *fw_buf_ptr
= NULL
;
244 uint64_t fw_offset
= 0;
246 struct amdgpu_firmware_info
*ucode
= NULL
;
247 const struct common_firmware_header
*header
= NULL
;
249 err
= amdgpu_bo_create(adev
, adev
->firmware
.fw_size
, PAGE_SIZE
, true,
250 AMDGPU_GEM_DOMAIN_GTT
, 0, NULL
, bo
);
252 dev_err(adev
->dev
, "(%d) Firmware buffer allocate failed\n", err
);
257 err
= amdgpu_bo_reserve(*bo
, false);
260 dev_err(adev
->dev
, "(%d) Firmware buffer reserve failed\n", err
);
264 err
= amdgpu_bo_pin(*bo
, AMDGPU_GEM_DOMAIN_GTT
, &fw_mc_addr
);
266 amdgpu_bo_unreserve(*bo
);
268 dev_err(adev
->dev
, "(%d) Firmware buffer pin failed\n", err
);
272 err
= amdgpu_bo_kmap(*bo
, &fw_buf_ptr
);
274 dev_err(adev
->dev
, "(%d) Firmware buffer kmap failed\n", err
);
275 amdgpu_bo_unpin(*bo
);
276 amdgpu_bo_unreserve(*bo
);
281 amdgpu_bo_unreserve(*bo
);
284 for (i
= 0; i
< AMDGPU_UCODE_ID_MAXIMUM
; i
++) {
285 ucode
= &adev
->firmware
.ucode
[i
];
287 header
= (const struct common_firmware_header
*)ucode
->fw
->data
;
288 amdgpu_ucode_init_single_fw(ucode
, fw_mc_addr
+ fw_offset
,
289 fw_buf_ptr
+ fw_offset
);
290 fw_offset
+= ALIGN(le32_to_cpu(header
->ucode_size_bytes
), PAGE_SIZE
);
296 adev
->firmware
.smu_load
= false;
301 int amdgpu_ucode_fini_bo(struct amdgpu_device
*adev
)
304 struct amdgpu_firmware_info
*ucode
= NULL
;
306 for (i
= 0; i
< AMDGPU_UCODE_ID_MAXIMUM
; i
++) {
307 ucode
= &adev
->firmware
.ucode
[i
];
313 amdgpu_bo_unref(&adev
->firmware
.fw_buf
);
314 adev
->firmware
.fw_buf
= NULL
;