[e16e8f2] | 1 | /* ----------------------------------------------------------------------- * |
---|
| 2 | * |
---|
| 3 | * Copyright 2009 Erwan Velu - All Rights Reserved |
---|
| 4 | * |
---|
| 5 | * Permission is hereby granted, free of charge, to any person |
---|
| 6 | * obtaining a copy of this software and associated documentation |
---|
| 7 | * files (the "Software"), to deal in the Software without |
---|
| 8 | * restriction, including without limitation the rights to use, |
---|
| 9 | * copy, modify, merge, publish, distribute, sublicense, and/or |
---|
| 10 | * sell copies of the Software, and to permit persons to whom |
---|
| 11 | * the Software is furnished to do so, subject to the following |
---|
| 12 | * conditions: |
---|
| 13 | * |
---|
| 14 | * The above copyright notice and this permission notice shall |
---|
| 15 | * be included in all copies or substantial portions of the Software. |
---|
| 16 | * |
---|
| 17 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
---|
| 18 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES |
---|
| 19 | * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND |
---|
| 20 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT |
---|
| 21 | * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, |
---|
| 22 | * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
---|
| 23 | * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR |
---|
| 24 | * OTHER DEALINGS IN THE SOFTWARE. |
---|
| 25 | * |
---|
| 26 | * ----------------------------------------------------------------------- |
---|
| 27 | */ |
---|
| 28 | |
---|
| 29 | #include <memory.h> |
---|
| 30 | #include "hdt-menu.h" |
---|
| 31 | #define E820MAX 128 |
---|
| 32 | |
---|
| 33 | /* Compute the e820 submenu */ |
---|
| 34 | static void compute_e820(struct s_my_menu *menu) |
---|
| 35 | { |
---|
| 36 | char buffer[MENULEN + 1]; |
---|
| 37 | char statbuffer[STATLEN + 1]; |
---|
| 38 | |
---|
| 39 | sprintf(buffer, " e820 information "); |
---|
| 40 | menu->items_count = 0; |
---|
| 41 | menu->menu = add_menu(buffer, -1); |
---|
| 42 | |
---|
| 43 | struct e820entry map[E820MAX]; |
---|
| 44 | int count = 0; |
---|
| 45 | char type[14]; |
---|
| 46 | |
---|
| 47 | detect_memory_e820(map, E820MAX, &count); |
---|
| 48 | unsigned long memory_size = memsize_e820(map, count); |
---|
| 49 | snprintf(buffer, sizeof buffer, "Detected Memory - %lu MiB (%lu KiB)", |
---|
| 50 | memory_size >> 10, memory_size); |
---|
| 51 | snprintf(statbuffer, sizeof statbuffer, |
---|
| 52 | "Detected Memory : %lu MiB (%lu KiB)", memory_size >> 10, |
---|
| 53 | memory_size); |
---|
| 54 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 55 | add_item("", "", OPT_SEP, "", 0); |
---|
| 56 | |
---|
| 57 | for (int j = 0; j < count; j++) { |
---|
| 58 | get_type(map[j].type, type, 14); |
---|
| 59 | snprintf(buffer, sizeof buffer, |
---|
| 60 | "%016llx - %016llx (%s)", |
---|
| 61 | map[j].addr, map[j].size, remove_spaces(type)); |
---|
| 62 | snprintf(statbuffer, sizeof statbuffer, |
---|
| 63 | "%016llx - %016llx (%s)", |
---|
| 64 | map[j].addr, map[j].size, remove_spaces(type)); |
---|
| 65 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 66 | menu->items_count++; |
---|
| 67 | } |
---|
| 68 | } |
---|
| 69 | |
---|
| 70 | /* Compute the e801 submenu */ |
---|
| 71 | static void compute_e801(struct s_my_menu *menu) |
---|
| 72 | { |
---|
| 73 | char buffer[MENULEN + 1]; |
---|
| 74 | char statbuffer[STATLEN + 1]; |
---|
| 75 | |
---|
| 76 | sprintf(buffer, " e801 information "); |
---|
| 77 | menu->items_count = 0; |
---|
| 78 | menu->menu = add_menu(buffer, -1); |
---|
| 79 | |
---|
| 80 | int mem_low, mem_high = 0; |
---|
| 81 | if (detect_memory_e801(&mem_low, &mem_high)) { |
---|
| 82 | snprintf(buffer, sizeof buffer, "%s", "e801 output is bogus"); |
---|
| 83 | snprintf(statbuffer, sizeof statbuffer, "%s", "e801 output is bogus"); |
---|
| 84 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 85 | menu->items_count++; |
---|
| 86 | } else { |
---|
| 87 | snprintf(buffer, sizeof buffer, "Detected Memory : %d MiB (%d KiB)", |
---|
| 88 | (mem_high >> 4) + (mem_low >> 10), mem_low + (mem_high << 6)); |
---|
| 89 | snprintf(statbuffer, sizeof statbuffer, |
---|
| 90 | "Detected Memory : %d MiB (%d KiB)", |
---|
| 91 | (mem_high >> 4) + (mem_low >> 10), mem_low + (mem_high << 6)); |
---|
| 92 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 93 | |
---|
| 94 | add_item("", "", OPT_SEP, "", 0); |
---|
| 95 | snprintf(buffer, sizeof buffer, "Low Memory : %d KiB (%d MiB)", |
---|
| 96 | mem_low, mem_low >> 10); |
---|
| 97 | snprintf(statbuffer, sizeof statbuffer, "Low Memory : %d KiB (%d MiB)", |
---|
| 98 | mem_low, mem_low >> 10); |
---|
| 99 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 100 | |
---|
| 101 | snprintf(buffer, sizeof buffer, "High Memory : %d KiB (%d MiB)", |
---|
| 102 | mem_high << 6, mem_high >> 4); |
---|
| 103 | snprintf(statbuffer, sizeof statbuffer, "High Memory : %d KiB (%d MiB)", |
---|
| 104 | mem_high << 6, mem_high >> 4); |
---|
| 105 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 106 | |
---|
| 107 | } |
---|
| 108 | menu->items_count++; |
---|
| 109 | } |
---|
| 110 | |
---|
| 111 | /* Compute the 88 submenu */ |
---|
| 112 | static void compute_88(struct s_my_menu *menu) |
---|
| 113 | { |
---|
| 114 | char buffer[MENULEN + 1]; |
---|
| 115 | char statbuffer[STATLEN + 1]; |
---|
| 116 | |
---|
| 117 | sprintf(buffer, " 88 information "); |
---|
| 118 | menu->items_count = 0; |
---|
| 119 | menu->menu = add_menu(buffer, -1); |
---|
| 120 | |
---|
| 121 | int mem_size = 0; |
---|
| 122 | if (detect_memory_88(&mem_size)) { |
---|
| 123 | snprintf(buffer, sizeof buffer, "%s", "88 output is bogus"); |
---|
| 124 | snprintf(statbuffer, sizeof statbuffer, "%s", "88 output is bogus"); |
---|
| 125 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 126 | menu->items_count++; |
---|
| 127 | } else { |
---|
| 128 | snprintf(buffer, sizeof buffer, "Detected Memory : %d MiB (%d KiB)", |
---|
| 129 | mem_size >> 10, mem_size); |
---|
| 130 | snprintf(statbuffer, sizeof statbuffer, |
---|
| 131 | "Detected Memory : %d MiB (%d KiB)", mem_size >> 10, mem_size); |
---|
| 132 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 133 | } |
---|
| 134 | menu->items_count++; |
---|
| 135 | } |
---|
| 136 | |
---|
| 137 | /* Compute the Memory submenu */ |
---|
| 138 | static void compute_memory_module(struct s_my_menu *menu, s_dmi * dmi, |
---|
| 139 | int slot_number) |
---|
| 140 | { |
---|
| 141 | int i = slot_number; |
---|
| 142 | char buffer[MENULEN + 1]; |
---|
| 143 | char statbuffer[STATLEN + 1]; |
---|
| 144 | |
---|
| 145 | sprintf(buffer, " Bank <%d> ", i); |
---|
| 146 | menu->items_count = 0; |
---|
| 147 | menu->menu = add_menu(buffer, -1); |
---|
| 148 | |
---|
| 149 | snprintf(buffer, sizeof buffer, "Form Factor : %s", |
---|
| 150 | dmi->memory[i].form_factor); |
---|
| 151 | snprintf(statbuffer, sizeof statbuffer, "Form Factor: %s", |
---|
| 152 | dmi->memory[i].form_factor); |
---|
| 153 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 154 | menu->items_count++; |
---|
| 155 | |
---|
| 156 | snprintf(buffer, sizeof buffer, "Type : %s", dmi->memory[i].type); |
---|
| 157 | snprintf(statbuffer, sizeof statbuffer, "Type: %s", dmi->memory[i].type); |
---|
| 158 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 159 | menu->items_count++; |
---|
| 160 | |
---|
| 161 | snprintf(buffer, sizeof buffer, "Type Details : %s", |
---|
| 162 | dmi->memory[i].type_detail); |
---|
| 163 | snprintf(statbuffer, sizeof statbuffer, "Type Details: %s", |
---|
| 164 | dmi->memory[i].type_detail); |
---|
| 165 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 166 | menu->items_count++; |
---|
| 167 | |
---|
| 168 | snprintf(buffer, sizeof buffer, "Speed : %s", dmi->memory[i].speed); |
---|
| 169 | snprintf(statbuffer, sizeof statbuffer, "Speed (Mhz): %s", |
---|
| 170 | dmi->memory[i].speed); |
---|
| 171 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 172 | menu->items_count++; |
---|
| 173 | |
---|
| 174 | snprintf(buffer, sizeof buffer, "Size : %s", dmi->memory[i].size); |
---|
| 175 | snprintf(statbuffer, sizeof statbuffer, "Size: %s", dmi->memory[i].size); |
---|
| 176 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 177 | menu->items_count++; |
---|
| 178 | |
---|
| 179 | snprintf(buffer, sizeof buffer, "Device Set : %s", |
---|
| 180 | dmi->memory[i].device_set); |
---|
| 181 | snprintf(statbuffer, sizeof statbuffer, "Device Set: %s", |
---|
| 182 | dmi->memory[i].device_set); |
---|
| 183 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 184 | menu->items_count++; |
---|
| 185 | |
---|
| 186 | snprintf(buffer, sizeof buffer, "Device Loc. : %s", |
---|
| 187 | dmi->memory[i].device_locator); |
---|
| 188 | snprintf(statbuffer, sizeof statbuffer, "Device Location: %s", |
---|
| 189 | dmi->memory[i].device_locator); |
---|
| 190 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 191 | menu->items_count++; |
---|
| 192 | |
---|
| 193 | snprintf(buffer, sizeof buffer, "Bank Locator : %s", |
---|
| 194 | dmi->memory[i].bank_locator); |
---|
| 195 | snprintf(statbuffer, sizeof statbuffer, "Bank Locator: %s", |
---|
| 196 | dmi->memory[i].bank_locator); |
---|
| 197 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 198 | menu->items_count++; |
---|
| 199 | |
---|
| 200 | snprintf(buffer, sizeof buffer, "Total Width : %s", |
---|
| 201 | dmi->memory[i].total_width); |
---|
| 202 | snprintf(statbuffer, sizeof statbuffer, "Total bit Width: %s", |
---|
| 203 | dmi->memory[i].total_width); |
---|
| 204 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 205 | menu->items_count++; |
---|
| 206 | |
---|
| 207 | snprintf(buffer, sizeof buffer, "Data Width : %s", |
---|
| 208 | dmi->memory[i].data_width); |
---|
| 209 | snprintf(statbuffer, sizeof statbuffer, "Data bit Width: %s", |
---|
| 210 | dmi->memory[i].data_width); |
---|
| 211 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 212 | menu->items_count++; |
---|
| 213 | |
---|
| 214 | snprintf(buffer, sizeof buffer, "Error : %s", dmi->memory[i].error); |
---|
| 215 | snprintf(statbuffer, sizeof statbuffer, "Error: %s", dmi->memory[i].error); |
---|
| 216 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 217 | menu->items_count++; |
---|
| 218 | |
---|
| 219 | snprintf(buffer, sizeof buffer, "Vendor : %s", |
---|
| 220 | dmi->memory[i].manufacturer); |
---|
| 221 | snprintf(statbuffer, sizeof statbuffer, "Vendor: %s", |
---|
| 222 | dmi->memory[i].manufacturer); |
---|
| 223 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 224 | menu->items_count++; |
---|
| 225 | |
---|
| 226 | snprintf(buffer, sizeof buffer, "Serial : %s", dmi->memory[i].serial); |
---|
| 227 | snprintf(statbuffer, sizeof statbuffer, "Serial: %s", |
---|
| 228 | dmi->memory[i].serial); |
---|
| 229 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 230 | menu->items_count++; |
---|
| 231 | |
---|
| 232 | snprintf(buffer, sizeof buffer, "Asset Tag : %s", |
---|
| 233 | dmi->memory[i].asset_tag); |
---|
| 234 | snprintf(statbuffer, sizeof statbuffer, "Asset Tag: %s", |
---|
| 235 | dmi->memory[i].asset_tag); |
---|
| 236 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 237 | menu->items_count++; |
---|
| 238 | |
---|
| 239 | snprintf(buffer, sizeof buffer, "Part Number : %s", |
---|
| 240 | dmi->memory[i].part_number); |
---|
| 241 | snprintf(statbuffer, sizeof statbuffer, "Part Number: %s", |
---|
| 242 | dmi->memory[i].part_number); |
---|
| 243 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 244 | menu->items_count++; |
---|
| 245 | |
---|
| 246 | } |
---|
| 247 | |
---|
| 248 | /* Compute the Memory submenu when type 6 is used*/ |
---|
| 249 | static void compute_memory_module_type6(struct s_my_menu *menu, s_dmi * dmi, |
---|
| 250 | int slot_number) |
---|
| 251 | { |
---|
| 252 | int i = slot_number; |
---|
| 253 | char buffer[MENULEN + 1]; |
---|
| 254 | char statbuffer[STATLEN + 1]; |
---|
| 255 | |
---|
| 256 | sprintf(buffer, " Bank <%d> ", i); |
---|
| 257 | menu->items_count = 0; |
---|
| 258 | menu->menu = add_menu(buffer, -1); |
---|
| 259 | |
---|
| 260 | snprintf(buffer, sizeof buffer, "Socket Designation : %s", |
---|
| 261 | dmi->memory_module[i].socket_designation); |
---|
| 262 | snprintf(statbuffer, sizeof statbuffer, "Socket Designation : %s", |
---|
| 263 | dmi->memory_module[i].socket_designation); |
---|
| 264 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 265 | menu->items_count++; |
---|
| 266 | |
---|
| 267 | snprintf(buffer, sizeof buffer, "Bank Connections : %s", |
---|
| 268 | dmi->memory_module[i].bank_connections); |
---|
| 269 | snprintf(statbuffer, sizeof statbuffer, "Bank Connections: %s", |
---|
| 270 | dmi->memory_module[i].bank_connections); |
---|
| 271 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 272 | menu->items_count++; |
---|
| 273 | |
---|
| 274 | snprintf(buffer, sizeof buffer, "Type : %s", |
---|
| 275 | dmi->memory_module[i].type); |
---|
| 276 | snprintf(statbuffer, sizeof statbuffer, "Type : %s", |
---|
| 277 | dmi->memory_module[i].type); |
---|
| 278 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 279 | menu->items_count++; |
---|
| 280 | |
---|
| 281 | snprintf(buffer, sizeof buffer, "Current Speed : %s", |
---|
| 282 | dmi->memory_module[i].speed); |
---|
| 283 | snprintf(statbuffer, sizeof statbuffer, "Current Speed : %s", |
---|
| 284 | dmi->memory_module[i].speed); |
---|
| 285 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 286 | menu->items_count++; |
---|
| 287 | |
---|
| 288 | snprintf(buffer, sizeof buffer, "Installed Size : %s", |
---|
| 289 | dmi->memory_module[i].installed_size); |
---|
| 290 | snprintf(statbuffer, sizeof statbuffer, "Installed Size : %s", |
---|
| 291 | dmi->memory_module[i].installed_size); |
---|
| 292 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 293 | menu->items_count++; |
---|
| 294 | |
---|
| 295 | snprintf(buffer, sizeof buffer, "Enabled Size : %s", |
---|
| 296 | dmi->memory_module[i].enabled_size); |
---|
| 297 | snprintf(statbuffer, sizeof statbuffer, "Enabled Size : %s", |
---|
| 298 | dmi->memory_module[i].enabled_size); |
---|
| 299 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 300 | menu->items_count++; |
---|
| 301 | |
---|
| 302 | snprintf(buffer, sizeof buffer, "Error Status : %s", |
---|
| 303 | dmi->memory_module[i].error_status); |
---|
| 304 | snprintf(statbuffer, sizeof statbuffer, "Error Status : %s", |
---|
| 305 | dmi->memory_module[i].error_status); |
---|
| 306 | add_item(buffer, statbuffer, OPT_INACTIVE, NULL, 0); |
---|
| 307 | menu->items_count++; |
---|
| 308 | |
---|
| 309 | } |
---|
| 310 | |
---|
| 311 | /* Compute the Memory menu */ |
---|
| 312 | void compute_memory(struct s_hdt_menu *menu, s_dmi * dmi, |
---|
| 313 | struct s_hardware *hardware) |
---|
| 314 | { |
---|
| 315 | char buffer[MENULEN + 1]; |
---|
| 316 | int i = 0; |
---|
| 317 | int memory_count = 0; |
---|
| 318 | |
---|
| 319 | /* If memory type 17 is available */ |
---|
| 320 | if (dmi->memory_count > 0) { |
---|
| 321 | memory_count = dmi->memory_count; |
---|
| 322 | for (i = 0; i < dmi->memory_count; i++) { |
---|
| 323 | compute_memory_module(&(menu->memory_sub_menu[i]), dmi, i); |
---|
| 324 | } |
---|
| 325 | } else if (dmi->memory_module_count > 0) { |
---|
| 326 | memory_count = dmi->memory_module_count; |
---|
| 327 | /* Memory Type 17 isn't available, let's fallback on type 6 */ |
---|
| 328 | for (i = 0; i < dmi->memory_module_count; i++) { |
---|
| 329 | compute_memory_module_type6(&(menu->memory_sub_menu[i]), dmi, i); |
---|
| 330 | } |
---|
| 331 | } |
---|
| 332 | |
---|
| 333 | compute_e820(&(menu->memory_sub_menu[++i])); |
---|
| 334 | compute_e801(&(menu->memory_sub_menu[++i])); |
---|
| 335 | compute_88(&(menu->memory_sub_menu[++i])); |
---|
| 336 | |
---|
| 337 | menu->memory_menu.menu = add_menu(" Memory ", -1); |
---|
| 338 | menu->memory_menu.items_count = 0; |
---|
| 339 | |
---|
| 340 | snprintf(buffer, sizeof(buffer), " %lu MB detected ", |
---|
| 341 | (hardware->detected_memory_size + (1 << 9)) >> 10); |
---|
| 342 | add_item(buffer, "Detected Memory", OPT_INACTIVE, NULL, |
---|
| 343 | menu->memory_sub_menu[0].menu); |
---|
| 344 | menu->memory_menu.items_count++; |
---|
| 345 | |
---|
| 346 | add_item("", "", OPT_SEP, "", 0); |
---|
| 347 | |
---|
| 348 | if (memory_count == 0) { |
---|
| 349 | snprintf(buffer, sizeof buffer, " No memory bank detected "); |
---|
| 350 | add_item(buffer, "Memory Bank", OPT_INACTIVE, NULL, |
---|
| 351 | menu->memory_sub_menu[1].menu); |
---|
| 352 | menu->memory_menu.items_count++; |
---|
| 353 | } else |
---|
| 354 | for (i = 0; i < memory_count; i++) { |
---|
| 355 | snprintf(buffer, sizeof buffer, " Bank <%d> ", i); |
---|
| 356 | add_item(buffer, "Memory Bank", OPT_SUBMENU, NULL, |
---|
| 357 | menu->memory_sub_menu[i].menu); |
---|
| 358 | menu->memory_menu.items_count++; |
---|
| 359 | } |
---|
| 360 | |
---|
| 361 | add_item("", "", OPT_SEP, "", 0); |
---|
| 362 | |
---|
| 363 | snprintf(buffer, sizeof buffer, " e820 "); |
---|
| 364 | add_item(buffer, "e820 mapping", OPT_SUBMENU, NULL, |
---|
| 365 | menu->memory_sub_menu[++i].menu); |
---|
| 366 | menu->memory_menu.items_count++; |
---|
| 367 | |
---|
| 368 | snprintf(buffer, sizeof buffer, " e801 "); |
---|
| 369 | add_item(buffer, "e801 information", OPT_SUBMENU, NULL, |
---|
| 370 | menu->memory_sub_menu[++i].menu); |
---|
| 371 | menu->memory_menu.items_count++; |
---|
| 372 | |
---|
| 373 | snprintf(buffer, sizeof buffer, " 88 "); |
---|
| 374 | add_item(buffer, "88 information", OPT_SUBMENU, NULL, |
---|
| 375 | menu->memory_sub_menu[++i].menu); |
---|
| 376 | menu->memory_menu.items_count++; |
---|
| 377 | |
---|
| 378 | add_item("", "", OPT_SEP, "", 0); |
---|
| 379 | printf("MENU: Memory menu done (%d items)\n", |
---|
| 380 | menu->memory_menu.items_count); |
---|
| 381 | add_item("Run Test", "Run Test", OPT_RUN, hardware->memtest_label, 0); |
---|
| 382 | } |
---|