head64.S 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. /*
  2. * arch/s390/kernel/head64.S
  3. *
  4. * (C) Copyright IBM Corp. 1999,2005
  5. *
  6. * Author(s): Hartmut Penner <hp@de.ibm.com>
  7. * Martin Schwidefsky <schwidefsky@de.ibm.com>
  8. * Rob van der Heij <rvdhei@iae.nl>
  9. * Heiko Carstens <heiko.carstens@de.ibm.com>
  10. *
  11. */
  12. #
  13. # startup-code at 0x10000, running in absolute addressing mode
  14. # this is called either by the ipl loader or directly by PSW restart
  15. # or linload or SALIPL
  16. #
  17. .org 0x10000
  18. startup:basr %r13,0 # get base
  19. .LPG1: sll %r13,1 # remove high order bit
  20. srl %r13,1
  21. l %r1,.Lget_ipl_device_addr-.LPG1(%r13)
  22. basr %r14,%r1
  23. lhi %r1,1 # mode 1 = esame
  24. slr %r0,%r0 # set cpuid to zero
  25. sigp %r1,%r0,0x12 # switch to esame mode
  26. sam64 # switch to 64 bit mode
  27. lctlg %c0,%c15,.Lctl-.LPG1(%r13) # load control registers
  28. larl %r12,_pstart # pointer to parameter area
  29. # move IPL device to lowcore
  30. mvc __LC_IPLDEV(4),IPL_DEVICE+4-PARMAREA(%r12)
  31. #
  32. # clear bss memory
  33. #
  34. larl %r2,__bss_start # start of bss segment
  35. larl %r3,_end # end of bss segment
  36. sgr %r3,%r2 # length of bss
  37. sgr %r4,%r4 #
  38. sgr %r5,%r5 # set src,length and pad to zero
  39. mvcle %r2,%r4,0 # clear mem
  40. jo .-4 # branch back, if not finish
  41. l %r2,.Lrcp-.LPG1(%r13) # Read SCP forced command word
  42. .Lservicecall:
  43. stosm .Lpmask-.LPG1(%r13),0x01 # authorize ext interrupts
  44. stctg %r0,%r0,.Lcr-.LPG1(%r13) # get cr0
  45. la %r1,0x200 # set bit 22
  46. og %r1,.Lcr-.LPG1(%r13) # or old cr0 with r1
  47. stg %r1,.Lcr-.LPG1(%r13)
  48. lctlg %r0,%r0,.Lcr-.LPG1(%r13) # load modified cr0
  49. mvc __LC_EXT_NEW_PSW(8),.Lpcmsk-.LPG1(%r13) # set postcall psw
  50. larl %r1,.Lsclph
  51. stg %r1,__LC_EXT_NEW_PSW+8 # set handler
  52. larl %r4,_pstart # %r4 is our index for sccb stuff
  53. la %r1,.Lsccb-PARMAREA(%r4) # our sccb
  54. .insn rre,0xb2200000,%r2,%r1 # service call
  55. ipm %r1
  56. srl %r1,28 # get cc code
  57. xr %r3,%r3
  58. chi %r1,3
  59. be .Lfchunk-.LPG1(%r13) # leave
  60. chi %r1,2
  61. be .Lservicecall-.LPG1(%r13)
  62. lpswe .Lwaitsclp-.LPG1(%r13)
  63. .Lsclph:
  64. lh %r1,.Lsccbr-PARMAREA(%r4)
  65. chi %r1,0x10 # 0x0010 is the sucess code
  66. je .Lprocsccb # let's process the sccb
  67. chi %r1,0x1f0
  68. bne .Lfchunk-.LPG1(%r13) # unhandled error code
  69. c %r2,.Lrcp-.LPG1(%r13) # Did we try Read SCP forced
  70. bne .Lfchunk-.LPG1(%r13) # if no, give up
  71. l %r2,.Lrcp2-.LPG1(%r13) # try with Read SCP
  72. b .Lservicecall-.LPG1(%r13)
  73. .Lprocsccb:
  74. lghi %r1,0
  75. icm %r1,3,.Lscpincr1-PARMAREA(%r4) # use this one if != 0
  76. jnz .Lscnd
  77. lg %r1,.Lscpincr2-PARMAREA(%r4) # otherwise use this one
  78. .Lscnd:
  79. xr %r3,%r3 # same logic
  80. ic %r3,.Lscpa1-PARMAREA(%r4)
  81. chi %r3,0x00
  82. jne .Lcompmem
  83. l %r3,.Lscpa2-PARMAREA(%r13)
  84. .Lcompmem:
  85. mlgr %r2,%r1 # mem in MB on 128-bit
  86. l %r1,.Lonemb-.LPG1(%r13)
  87. mlgr %r2,%r1 # mem size in bytes in %r3
  88. b .Lfchunk-.LPG1(%r13)
  89. .align 4
  90. .Lget_ipl_device_addr:
  91. .long .Lget_ipl_device
  92. .Lpmask:
  93. .byte 0
  94. .align 8
  95. .Lcr:
  96. .quad 0x00 # place holder for cr0
  97. .Lwaitsclp:
  98. .quad 0x0102000180000000,.Lsclph
  99. .Lrcp:
  100. .int 0x00120001 # Read SCP forced code
  101. .Lrcp2:
  102. .int 0x00020001 # Read SCP code
  103. .Lonemb:
  104. .int 0x100000
  105. .Lfchunk:
  106. # set program check new psw mask
  107. mvc __LC_PGM_NEW_PSW(8),.Lpcmsk-.LPG1(%r13)
  108. #
  109. # find memory chunks.
  110. #
  111. lgr %r9,%r3 # end of mem
  112. larl %r1,.Lchkmem # set program check address
  113. stg %r1,__LC_PGM_NEW_PSW+8
  114. la %r1,1 # test in increments of 128KB
  115. sllg %r1,%r1,17
  116. larl %r3,memory_chunk
  117. slgr %r4,%r4 # set start of chunk to zero
  118. slgr %r5,%r5 # set end of chunk to zero
  119. slr %r6,%r6 # set access code to zero
  120. la %r10,MEMORY_CHUNKS # number of chunks
  121. .Lloop:
  122. tprot 0(%r5),0 # test protection of first byte
  123. ipm %r7
  124. srl %r7,28
  125. clr %r6,%r7 # compare cc with last access code
  126. je .Lsame
  127. j .Lchkmem
  128. .Lsame:
  129. algr %r5,%r1 # add 128KB to end of chunk
  130. # no need to check here,
  131. brc 12,.Lloop # this is the same chunk
  132. .Lchkmem: # > 16EB or tprot got a program check
  133. clgr %r4,%r5 # chunk size > 0?
  134. je .Lchkloop
  135. stg %r4,0(%r3) # store start address of chunk
  136. lgr %r0,%r5
  137. slgr %r0,%r4
  138. stg %r0,8(%r3) # store size of chunk
  139. st %r6,20(%r3) # store type of chunk
  140. la %r3,24(%r3)
  141. larl %r8,memory_size
  142. stg %r5,0(%r8) # store memory size
  143. ahi %r10,-1 # update chunk number
  144. .Lchkloop:
  145. lr %r6,%r7 # set access code to last cc
  146. # we got an exception or we're starting a new
  147. # chunk , we must check if we should
  148. # still try to find valid memory (if we detected
  149. # the amount of available storage), and if we
  150. # have chunks left
  151. lghi %r4,1
  152. sllg %r4,%r4,31
  153. clgr %r5,%r4
  154. je .Lhsaskip
  155. xr %r0, %r0
  156. clgr %r0, %r9 # did we detect memory?
  157. je .Ldonemem # if not, leave
  158. chi %r10, 0 # do we have chunks left?
  159. je .Ldonemem
  160. .Lhsaskip:
  161. algr %r5,%r1 # add 128KB to end of chunk
  162. lgr %r4,%r5 # potential new chunk
  163. clgr %r5,%r9 # should we go on?
  164. jl .Lloop
  165. .Ldonemem:
  166. larl %r12,machine_flags
  167. #
  168. # find out if we are running under VM
  169. #
  170. stidp __LC_CPUID # store cpuid
  171. tm __LC_CPUID,0xff # running under VM ?
  172. bno 0f-.LPG1(%r13)
  173. oi 7(%r12),1 # set VM flag
  174. 0: lh %r0,__LC_CPUID+4 # get cpu version
  175. chi %r0,0x7490 # running on a P/390 ?
  176. bne 1f-.LPG1(%r13)
  177. oi 7(%r12),4 # set P/390 flag
  178. 1:
  179. #
  180. # find out if we have the MVPG instruction
  181. #
  182. la %r1,0f-.LPG1(%r13) # set program check address
  183. stg %r1,__LC_PGM_NEW_PSW+8
  184. sgr %r0,%r0
  185. lghi %r1,0
  186. lghi %r2,0
  187. mvpg %r1,%r2 # test MVPG instruction
  188. oi 7(%r12),16 # set MVPG flag
  189. 0:
  190. #
  191. # find out if the diag 0x44 works in 64 bit mode
  192. #
  193. la %r1,0f-.LPG1(%r13) # set program check address
  194. stg %r1,__LC_PGM_NEW_PSW+8
  195. diag 0,0,0x44 # test diag 0x44
  196. oi 7(%r12),32 # set diag44 flag
  197. 0:
  198. #
  199. # find out if we have the IDTE instruction
  200. #
  201. la %r1,0f-.LPG1(%r13) # set program check address
  202. stg %r1,__LC_PGM_NEW_PSW+8
  203. .long 0xb2b10000 # store facility list
  204. tm 0xc8,0x08 # check bit for clearing-by-ASCE
  205. bno 0f-.LPG1(%r13)
  206. lhi %r1,2094
  207. lhi %r2,0
  208. .long 0xb98e2001
  209. oi 7(%r12),0x80 # set IDTE flag
  210. 0:
  211. lpswe .Lentry-.LPG1(13) # jump to _stext in primary-space,
  212. # virtual and never return ...
  213. .align 16
  214. .Lentry:.quad 0x0000000180000000,_stext
  215. .Lctl: .quad 0x04b50002 # cr0: various things
  216. .quad 0 # cr1: primary space segment table
  217. .quad .Lduct # cr2: dispatchable unit control table
  218. .quad 0 # cr3: instruction authorization
  219. .quad 0 # cr4: instruction authorization
  220. .quad 0xffffffffffffffff # cr5: primary-aste origin
  221. .quad 0 # cr6: I/O interrupts
  222. .quad 0 # cr7: secondary space segment table
  223. .quad 0 # cr8: access registers translation
  224. .quad 0 # cr9: tracing off
  225. .quad 0 # cr10: tracing off
  226. .quad 0 # cr11: tracing off
  227. .quad 0 # cr12: tracing off
  228. .quad 0 # cr13: home space segment table
  229. .quad 0xc0000000 # cr14: machine check handling off
  230. .quad 0 # cr15: linkage stack operations
  231. .Lpcmsk:.quad 0x0000000180000000
  232. .L4malign:.quad 0xffffffffffc00000
  233. .Lscan2g:.quad 0x80000000 + 0x20000 - 8 # 2GB + 128K - 8
  234. .Lnop: .long 0x07000700
  235. .org PARMAREA-64
  236. .Lduct: .long 0,0,0,0,0,0,0,0
  237. .long 0,0,0,0,0,0,0,0
  238. #
  239. # params at 10400 (setup.h)
  240. #
  241. .org PARMAREA
  242. .global _pstart
  243. _pstart:
  244. .quad 0 # IPL_DEVICE
  245. .quad RAMDISK_ORIGIN # INITRD_START
  246. .quad RAMDISK_SIZE # INITRD_SIZE
  247. .org COMMAND_LINE
  248. .byte "root=/dev/ram0 ro"
  249. .byte 0
  250. .org 0x11000
  251. .Lsccb:
  252. .hword 0x1000 # length, one page
  253. .byte 0x00,0x00,0x00
  254. .byte 0x80 # variable response bit set
  255. .Lsccbr:
  256. .hword 0x00 # response code
  257. .Lscpincr1:
  258. .hword 0x00
  259. .Lscpa1:
  260. .byte 0x00
  261. .fill 89,1,0
  262. .Lscpa2:
  263. .int 0x00
  264. .Lscpincr2:
  265. .quad 0x00
  266. .fill 3984,1,0
  267. .org 0x12000
  268. .global _pend
  269. _pend:
  270. GET_IPL_DEVICE
  271. #ifdef CONFIG_SHARED_KERNEL
  272. .org 0x100000
  273. #endif
  274. #
  275. # startup-code, running in virtual mode
  276. #
  277. .globl _stext
  278. _stext: basr %r13,0 # get base
  279. .LPG3:
  280. #
  281. # Setup stack
  282. #
  283. larl %r15,init_thread_union
  284. lg %r14,__TI_task(%r15) # cache current in lowcore
  285. stg %r14,__LC_CURRENT
  286. aghi %r15,1<<(PAGE_SHIFT+THREAD_ORDER) # init_task_union + THREAD_SIZE
  287. stg %r15,__LC_KERNEL_STACK # set end of kernel stack
  288. aghi %r15,-160
  289. xc __SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15) # clear backchain
  290. # check control registers
  291. stctg %c0,%c15,0(%r15)
  292. oi 6(%r15),0x40 # enable sigp emergency signal
  293. oi 4(%r15),0x10 # switch on low address proctection
  294. lctlg %c0,%c15,0(%r15)
  295. #
  296. lam 0,15,.Laregs-.LPG3(%r13) # load access regs needed by uaccess
  297. brasl %r14,start_kernel # go to C code
  298. #
  299. # We returned from start_kernel ?!? PANIK
  300. #
  301. basr %r13,0
  302. lpswe .Ldw-.(%r13) # load disabled wait psw
  303. #
  304. .align 8
  305. .Ldw: .quad 0x0002000180000000,0x0000000000000000
  306. .Laregs: .long 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0