cmd_mtrr.c 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. /*
  2. * (C) Copyright 2014 Google, Inc
  3. *
  4. * SPDX-License-Identifier: GPL-2.0+
  5. */
  6. #include <common.h>
  7. #include <asm/msr.h>
  8. #include <asm/mtrr.h>
  9. static const char *const mtrr_type_name[MTRR_TYPE_COUNT] = {
  10. "Uncacheable",
  11. "Combine",
  12. "2",
  13. "3",
  14. "Through",
  15. "Protect",
  16. "Back",
  17. };
  18. static int do_mtrr_list(void)
  19. {
  20. int i;
  21. printf("Reg Valid Write-type %-16s %-16s %-16s\n", "Base ||",
  22. "Mask ||", "Size ||");
  23. for (i = 0; i < MTRR_COUNT; i++) {
  24. const char *type = "Invalid";
  25. uint64_t base, mask, size;
  26. bool valid;
  27. base = native_read_msr(MTRR_PHYS_BASE_MSR(i));
  28. mask = native_read_msr(MTRR_PHYS_MASK_MSR(i));
  29. size = ~mask & ((1ULL << CONFIG_CPU_ADDR_BITS) - 1);
  30. size |= (1 << 12) - 1;
  31. size += 1;
  32. valid = mask & MTRR_PHYS_MASK_VALID;
  33. type = mtrr_type_name[base & MTRR_BASE_TYPE_MASK];
  34. printf("%d %-5s %-12s %016llx %016llx %016llx\n", i,
  35. valid ? "Y" : "N", type, base & ~MTRR_BASE_TYPE_MASK,
  36. mask & ~MTRR_PHYS_MASK_VALID, size);
  37. }
  38. return 0;
  39. }
  40. static int do_mtrr_set(uint reg, int argc, char * const argv[])
  41. {
  42. const char *typename = argv[0];
  43. struct mtrr_state state;
  44. uint32_t start, size;
  45. uint64_t base, mask;
  46. int i, type = -1;
  47. bool valid;
  48. if (argc < 3)
  49. return CMD_RET_USAGE;
  50. for (i = 0; i < MTRR_TYPE_COUNT; i++) {
  51. if (*typename == *mtrr_type_name[i])
  52. type = i;
  53. }
  54. if (type == -1) {
  55. printf("Invalid type name %s\n", typename);
  56. return CMD_RET_USAGE;
  57. }
  58. start = simple_strtoul(argv[1], NULL, 16);
  59. size = simple_strtoul(argv[2], NULL, 16);
  60. base = start | type;
  61. valid = native_read_msr(MTRR_PHYS_MASK_MSR(reg)) & MTRR_PHYS_MASK_VALID;
  62. mask = ~((uint64_t)size - 1);
  63. mask &= (1ULL << CONFIG_CPU_ADDR_BITS) - 1;
  64. if (valid)
  65. mask |= MTRR_PHYS_MASK_VALID;
  66. printf("base=%llx, mask=%llx\n", base, mask);
  67. mtrr_open(&state);
  68. wrmsrl(MTRR_PHYS_BASE_MSR(reg), base);
  69. wrmsrl(MTRR_PHYS_MASK_MSR(reg), mask);
  70. mtrr_close(&state);
  71. return 0;
  72. }
  73. static int mtrr_set_valid(int reg, bool valid)
  74. {
  75. struct mtrr_state state;
  76. uint64_t mask;
  77. mtrr_open(&state);
  78. mask = native_read_msr(MTRR_PHYS_MASK_MSR(reg));
  79. if (valid)
  80. mask |= MTRR_PHYS_MASK_VALID;
  81. else
  82. mask &= ~MTRR_PHYS_MASK_VALID;
  83. wrmsrl(MTRR_PHYS_MASK_MSR(reg), mask);
  84. mtrr_close(&state);
  85. return 0;
  86. }
  87. static int do_mtrr(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  88. {
  89. const char *cmd;
  90. uint reg;
  91. cmd = argv[1];
  92. if (argc < 2 || *cmd == 'l')
  93. return do_mtrr_list();
  94. argc -= 2;
  95. argv += 2;
  96. if (argc <= 0)
  97. return CMD_RET_USAGE;
  98. reg = simple_strtoul(argv[0], NULL, 16);
  99. if (reg >= MTRR_COUNT) {
  100. printf("Invalid register number\n");
  101. return CMD_RET_USAGE;
  102. }
  103. if (*cmd == 'e')
  104. return mtrr_set_valid(reg, true);
  105. else if (*cmd == 'd')
  106. return mtrr_set_valid(reg, false);
  107. else if (*cmd == 's')
  108. return do_mtrr_set(reg, argc - 1, argv + 1);
  109. else
  110. return CMD_RET_USAGE;
  111. return 0;
  112. }
  113. U_BOOT_CMD(
  114. mtrr, 6, 1, do_mtrr,
  115. "Use x86 memory type range registers (32-bit only)",
  116. "[list] - list current registers\n"
  117. "set <reg> <type> <start> <size> - set a register\n"
  118. "\t<type> is Uncacheable, Combine, Through, Protect, Back\n"
  119. "disable <reg> - disable a register\n"
  120. "ensable <reg> - enable a register"
  121. );