head64.S 9.6 KB

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