ipl_data.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/ptrace.h>
  3. #include <asm/cio.h>
  4. #include <asm/asm-offsets.h>
  5. #include "boot.h"
  6. #define CCW0(cmd, addr, cnt, flg) \
  7. { .cmd_code = cmd, .cda = addr, .count = cnt, .flags = flg, }
  8. #define PSW_MASK_DISABLED (PSW_MASK_WAIT | PSW_MASK_EA | PSW_MASK_BA)
  9. struct ipl_lowcore {
  10. psw32_t ipl_psw; /* 0x0000 */
  11. struct ccw0 ccwpgm[2]; /* 0x0008 */
  12. u8 fill[56]; /* 0x0018 */
  13. struct ccw0 ccwpgmcc[20]; /* 0x0050 */
  14. u8 pad_0xf0[0x0140-0x00f0]; /* 0x00f0 */
  15. psw_t svc_old_psw; /* 0x0140 */
  16. u8 pad_0x150[0x01a0-0x0150]; /* 0x0150 */
  17. psw_t restart_psw; /* 0x01a0 */
  18. psw_t external_new_psw; /* 0x01b0 */
  19. psw_t svc_new_psw; /* 0x01c0 */
  20. psw_t program_new_psw; /* 0x01d0 */
  21. psw_t mcck_new_psw; /* 0x01e0 */
  22. psw_t io_new_psw; /* 0x01f0 */
  23. };
  24. /*
  25. * Initial lowcore for IPL: the first 24 bytes are loaded by IPL to
  26. * addresses 0-23 (a PSW and two CCWs). Bytes 24-79 are discarded.
  27. * The next 160 bytes are loaded to addresses 0x18-0xb7. They form
  28. * the continuation of the CCW program started by IPL and load the
  29. * range 0x0f0-0x730 from the image to the range 0x0f0-0x730 in
  30. * memory. At the end of the channel program the PSW at location 0 is
  31. * loaded.
  32. * Initial processing starts at 0x200 = iplstart.
  33. *
  34. * The restart psw points to iplstart which allows to load a kernel
  35. * image into memory and starting it by a psw restart on any cpu. All
  36. * other default psw new locations contain a disabled wait psw where
  37. * the address indicates which psw was loaded.
  38. *
  39. * Note that the 'file' utility can detect s390 kernel images. For
  40. * that to succeed the two initial CCWs, and the 0x40 fill bytes must
  41. * be present.
  42. */
  43. static struct ipl_lowcore ipl_lowcore __used __section(".ipldata") = {
  44. .ipl_psw = { .mask = PSW32_MASK_BASE, .addr = PSW32_ADDR_AMODE | IPL_START },
  45. .ccwpgm = {
  46. [ 0] = CCW0(CCW_CMD_READ_IPL, 0x018, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  47. [ 1] = CCW0(CCW_CMD_READ_IPL, 0x068, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  48. },
  49. .fill = {
  50. [ 0 ... 55] = 0x40,
  51. },
  52. .ccwpgmcc = {
  53. [ 0] = CCW0(CCW_CMD_READ_IPL, 0x0f0, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  54. [ 1] = CCW0(CCW_CMD_READ_IPL, 0x140, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  55. [ 2] = CCW0(CCW_CMD_READ_IPL, 0x190, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  56. [ 3] = CCW0(CCW_CMD_READ_IPL, 0x1e0, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  57. [ 4] = CCW0(CCW_CMD_READ_IPL, 0x230, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  58. [ 5] = CCW0(CCW_CMD_READ_IPL, 0x280, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  59. [ 6] = CCW0(CCW_CMD_READ_IPL, 0x2d0, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  60. [ 7] = CCW0(CCW_CMD_READ_IPL, 0x320, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  61. [ 8] = CCW0(CCW_CMD_READ_IPL, 0x370, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  62. [ 9] = CCW0(CCW_CMD_READ_IPL, 0x3c0, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  63. [10] = CCW0(CCW_CMD_READ_IPL, 0x410, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  64. [11] = CCW0(CCW_CMD_READ_IPL, 0x460, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  65. [12] = CCW0(CCW_CMD_READ_IPL, 0x4b0, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  66. [13] = CCW0(CCW_CMD_READ_IPL, 0x500, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  67. [14] = CCW0(CCW_CMD_READ_IPL, 0x550, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  68. [15] = CCW0(CCW_CMD_READ_IPL, 0x5a0, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  69. [16] = CCW0(CCW_CMD_READ_IPL, 0x5f0, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  70. [17] = CCW0(CCW_CMD_READ_IPL, 0x640, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  71. [18] = CCW0(CCW_CMD_READ_IPL, 0x690, 0x50, CCW_FLAG_SLI | CCW_FLAG_CC),
  72. [19] = CCW0(CCW_CMD_READ_IPL, 0x6e0, 0x50, CCW_FLAG_SLI),
  73. },
  74. /*
  75. * Let the GDB's lx-symbols command find the jump_to_kernel symbol
  76. * without having to load decompressor symbols.
  77. */
  78. .svc_old_psw = { .mask = 0, .addr = (unsigned long)jump_to_kernel },
  79. .restart_psw = { .mask = 0, .addr = IPL_START, },
  80. .external_new_psw = { .mask = PSW_MASK_DISABLED, .addr = __LC_EXT_NEW_PSW, },
  81. .svc_new_psw = { .mask = PSW_MASK_DISABLED, .addr = __LC_SVC_NEW_PSW, },
  82. .program_new_psw = { .mask = PSW_MASK_DISABLED, .addr = __LC_PGM_NEW_PSW, },
  83. .mcck_new_psw = { .mask = PSW_MASK_DISABLED, .addr = __LC_MCK_NEW_PSW, },
  84. .io_new_psw = { .mask = PSW_MASK_DISABLED, .addr = __LC_IO_NEW_PSW, },
  85. };