You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

dgemm_kernel_4x2_vfp.S 13 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826
  1. /***************************************************************************
  2. Copyright (c) 2013, The OpenBLAS Project
  3. All rights reserved.
  4. Redistribution and use in source and binary forms, with or without
  5. modification, are permitted provided that the following conditions are
  6. met:
  7. 1. Redistributions of source code must retain the above copyright
  8. notice, this list of conditions and the following disclaimer.
  9. 2. Redistributions in binary form must reproduce the above copyright
  10. notice, this list of conditions and the following disclaimer in
  11. the documentation and/or other materials provided with the
  12. distribution.
  13. 3. Neither the name of the OpenBLAS project nor the names of
  14. its contributors may be used to endorse or promote products
  15. derived from this software without specific prior written permission.
  16. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  17. AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  18. IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  19. ARE DISCLAIMED. IN NO EVENT SHALL THE OPENBLAS PROJECT OR CONTRIBUTORS BE
  20. LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  21. DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
  22. SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  23. CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  24. OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
  25. USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  26. *****************************************************************************/
  27. /**************************************************************************************
  28. * 2013/11/27 Saar
  29. * BLASTEST : OK
  30. * CTEST : OK
  31. * TEST : OK
  32. *
  33. **************************************************************************************/
  34. #define ASSEMBLER
  35. #include "common.h"
  36. #define STACKSIZE 256
  37. #define OLD_M r0
  38. #define OLD_N r1
  39. #define OLD_K r2
  40. #define OLD_A r3
  41. #define OLD_ALPHA d0
  42. /******************************************************
  43. * [fp, #-128] - [fp, #-64] is reserved
  44. * for store and restore of floating point
  45. * registers
  46. *******************************************************/
  47. #define LDC [fp, #-252 ]
  48. #define M [fp, #-256 ]
  49. #define N [fp, #-260 ]
  50. #define K [fp, #-264 ]
  51. #define A [fp, #-268 ]
  52. #define FP_ZERO [fp, #-240]
  53. #define FP_ZERO_0 [fp, # -240]
  54. #define FP_ZERO_1 [fp, # -236]
  55. #define ALPHA [fp, #-280]
  56. #if !defined(__ARM_PCS_VFP)
  57. #define OLD_ALPHA_SOFTFP [fp, #4]
  58. #define OLD_A_SOFTFP [fp, #12 ]
  59. #define B [fp, #16 ]
  60. #define C [fp, #20 ]
  61. #define OLD_LDC [fp, #24 ]
  62. #else
  63. #define B [fp, #4 ]
  64. #define C [fp, #8 ]
  65. #define OLD_LDC [fp, #12 ]
  66. #endif
  67. #define I r0
  68. #define J r1
  69. #define L r2
  70. #define AO r5
  71. #define BO r6
  72. #define CO1 r8
  73. #define CO2 r9
  74. #define K1 r7
  75. #define BC r12
  76. #define A_PRE 96
  77. #define B_PRE 96
  78. #define C_PRE 32
  79. /**************************************************************************************
  80. * Macro definitions
  81. **************************************************************************************/
  82. .macro INIT4x2
  83. fldd d8, FP_ZERO
  84. vmov.f64 d9, d8
  85. vmov.f64 d10, d8
  86. vmov.f64 d11, d8
  87. vmov.f64 d12, d8
  88. vmov.f64 d13, d8
  89. vmov.f64 d14, d8
  90. vmov.f64 d15, d8
  91. .endm
  92. .macro KERNEL4x2_SUB
  93. pld [ AO, #A_PRE ]
  94. fldd d4 , [ BO ]
  95. fldd d0 , [ AO ]
  96. fldd d1 , [ AO, #8 ]
  97. fmacd d8 , d0, d4
  98. fldd d2 , [ AO, #16 ]
  99. fmacd d9 , d1, d4
  100. fldd d3 , [ AO, #24 ]
  101. fmacd d10 , d2, d4
  102. fldd d5 , [ BO, #8 ]
  103. fmacd d11 , d3, d4
  104. fmacd d12 , d0, d5
  105. fmacd d13 , d1, d5
  106. add AO , AO, #32
  107. fmacd d14 , d2, d5
  108. add BO , BO, #16
  109. fmacd d15 , d3, d5
  110. .endm
  111. .macro SAVE4x2
  112. ldr r3 , LDC
  113. add CO2 , CO1, r3
  114. fldd d0, ALPHA
  115. fldd d4 , [CO1]
  116. fldd d5 , [CO1, #8 ]
  117. pld [ CO1, #C_PRE ]
  118. fmacd d4 , d0 , d8
  119. fldd d6 , [CO1, #16 ]
  120. fmacd d5 , d0 , d9
  121. fldd d7 , [CO1, #24 ]
  122. fmacd d6 , d0 , d10
  123. fstd d4 , [CO1]
  124. fmacd d7 , d0 , d11
  125. fstd d5 , [CO1, #8 ]
  126. fstd d6 , [CO1, #16 ]
  127. fstd d7 , [CO1, #24 ]
  128. fldd d4 , [CO2]
  129. fldd d5 , [CO2, #8 ]
  130. pld [ CO2, #C_PRE ]
  131. fmacd d4 , d0 , d12
  132. fldd d6 , [CO2, #16 ]
  133. fmacd d5 , d0 , d13
  134. fldd d7 , [CO2, #24 ]
  135. fmacd d6 , d0 , d14
  136. fstd d4 , [CO2]
  137. fmacd d7 , d0 , d15
  138. add CO1, CO1, #32
  139. fstd d5 , [CO2, #8 ]
  140. fstd d6 , [CO2, #16 ]
  141. fstd d7 , [CO2, #24 ]
  142. .endm
  143. /******************************************************************************/
  144. .macro INIT2x2
  145. fldd d8, FP_ZERO
  146. vmov.f64 d9, d8
  147. vmov.f64 d12, d8
  148. vmov.f64 d13, d8
  149. .endm
  150. .macro KERNEL2x2_SUB
  151. fldd d4 , [ BO ]
  152. fldd d5 , [ BO, #8 ]
  153. fldd d0 , [ AO ]
  154. fldd d1 , [ AO, #8 ]
  155. fmacd d8 , d0, d4
  156. fmacd d9 , d1, d4
  157. fmacd d12 , d0, d5
  158. fmacd d13 , d1, d5
  159. add AO , AO, #16
  160. add BO , BO, #16
  161. .endm
  162. .macro SAVE2x2
  163. ldr r3 , LDC
  164. add CO2 , CO1, r3
  165. fldd d0, ALPHA
  166. fldd d4 , [CO1]
  167. fldd d5 , [CO1, #8 ]
  168. fmacd d4 , d0 , d8
  169. fmacd d5 , d0 , d9
  170. fstd d4 , [CO1]
  171. fstd d5 , [CO1, #8 ]
  172. fldd d4 , [CO2]
  173. fldd d5 , [CO2, #8 ]
  174. fmacd d4 , d0 , d12
  175. fmacd d5 , d0 , d13
  176. fstd d4 , [CO2]
  177. fstd d5 , [CO2, #8 ]
  178. add CO1, CO1, #16
  179. .endm
  180. /******************************************************************************/
  181. .macro INIT1x2
  182. fldd d8, FP_ZERO
  183. vmov.f64 d12, d8
  184. .endm
  185. .macro KERNEL1x2_SUB
  186. fldd d4 , [ BO ]
  187. fldd d5 , [ BO, #8 ]
  188. fldd d0 , [ AO ]
  189. fmacd d8 , d0, d4
  190. fmacd d12 , d0, d5
  191. add AO , AO, #8
  192. add BO , BO, #16
  193. .endm
  194. .macro SAVE1x2
  195. ldr r3 , LDC
  196. add CO2 , CO1, r3
  197. fldd d0, ALPHA
  198. fldd d4 , [CO1]
  199. fmacd d4 , d0 , d8
  200. fstd d4 , [CO1]
  201. fldd d4 , [CO2]
  202. fmacd d4 , d0 , d12
  203. fstd d4 , [CO2]
  204. add CO1, CO1, #8
  205. .endm
  206. /******************************************************************************/
  207. .macro INIT4x1
  208. fldd d8, FP_ZERO
  209. vmov.f64 d9, d8
  210. vmov.f64 d10, d8
  211. vmov.f64 d11, d8
  212. .endm
  213. .macro KERNEL4x1_SUB
  214. fldd d4 , [ BO ]
  215. fldd d0 , [ AO ]
  216. fldd d1 , [ AO, #8 ]
  217. fldd d2 , [ AO, #16 ]
  218. fldd d3 , [ AO, #24 ]
  219. fmacd d8 , d0, d4
  220. fmacd d9 , d1, d4
  221. fmacd d10 , d2, d4
  222. fmacd d11 , d3, d4
  223. add AO , AO, #32
  224. add BO , BO, #8
  225. .endm
  226. .macro SAVE4x1
  227. fldd d0, ALPHA
  228. fldd d4 , [CO1]
  229. fldd d5 , [CO1, #8 ]
  230. fldd d6 , [CO1, #16 ]
  231. fldd d7 , [CO1, #24 ]
  232. fmacd d4 , d0 , d8
  233. fmacd d5 , d0 , d9
  234. fmacd d6 , d0 , d10
  235. fmacd d7 , d0 , d11
  236. fstd d4 , [CO1]
  237. fstd d5 , [CO1, #8 ]
  238. fstd d6 , [CO1, #16 ]
  239. fstd d7 , [CO1, #24 ]
  240. add CO1, CO1, #32
  241. .endm
  242. /******************************************************************************/
  243. .macro INIT2x1
  244. fldd d8, FP_ZERO
  245. vmov.f64 d9 , d8
  246. .endm
  247. .macro KERNEL2x1_SUB
  248. fldd d4 , [ BO ]
  249. fldd d0 , [ AO ]
  250. fldd d1 , [ AO, #8 ]
  251. fmacd d8 , d0, d4
  252. fmacd d9 , d1, d4
  253. add AO , AO, #16
  254. add BO , BO, #8
  255. .endm
  256. .macro SAVE2x1
  257. fldd d0, ALPHA
  258. fldd d4 , [CO1]
  259. fldd d5 , [CO1, #8 ]
  260. fmacd d4 , d0 , d8
  261. fmacd d5 , d0 , d9
  262. fstd d4 , [CO1]
  263. fstd d5 , [CO1, #8 ]
  264. add CO1, CO1, #16
  265. .endm
  266. /******************************************************************************/
  267. .macro INIT1x1
  268. fldd d8, FP_ZERO
  269. .endm
  270. .macro KERNEL1x1_SUB
  271. fldd d4 , [ BO ]
  272. fldd d0 , [ AO ]
  273. fmacd d8 , d0, d4
  274. add AO , AO, #8
  275. add BO , BO, #8
  276. .endm
  277. .macro SAVE1x1
  278. fldd d0, ALPHA
  279. fldd d4 , [CO1]
  280. fmacd d4 , d0 , d8
  281. fstd d4 , [CO1]
  282. add CO1, CO1, #8
  283. .endm
  284. /**************************************************************************************
  285. * End of macro definitions
  286. **************************************************************************************/
  287. PROLOGUE
  288. .align 5
  289. push {r4 - r9, fp}
  290. add fp, sp, #24
  291. sub sp, sp, #STACKSIZE // reserve stack
  292. #if !defined(__ARM_PCS_VFP)
  293. vldr OLD_ALPHA, OLD_ALPHA_SOFTFP
  294. ldr OLD_A, OLD_A_SOFTFP
  295. #endif
  296. str OLD_M, M
  297. str OLD_N, N
  298. str OLD_K, K
  299. str OLD_A, A
  300. vstr OLD_ALPHA, ALPHA
  301. sub r3, fp, #128
  302. vstm r3, { d8 - d15} // store floating point registers
  303. movs r4, #0
  304. str r4, FP_ZERO
  305. str r4, FP_ZERO_1
  306. ldr r3, OLD_LDC
  307. lsl r3, r3, #3 // ldc = ldc * 8
  308. str r3, LDC
  309. ldr K1, K
  310. ldr BC, B
  311. ldr J, N
  312. asrs J, J, #1 // J = J / 2
  313. ble dgemm_kernel_L1_BEGIN
  314. /*********************************************************************************************/
  315. dgemm_kernel_L2_BEGIN:
  316. ldr CO1, C // CO1 = C
  317. ldr r4 , LDC
  318. lsl r4 , r4 , #1 // LDC * 2
  319. add r3 , r4, CO1
  320. str r3 , C // store C
  321. ldr AO, A // AO = A
  322. dgemm_kernel_L2_M4_BEGIN:
  323. ldr I, M
  324. asrs I, I, #2 // I = I / 4
  325. ble dgemm_kernel_L2_M2_BEGIN
  326. dgemm_kernel_L2_M4_20:
  327. INIT4x2
  328. mov BO, BC
  329. asrs L , K1, #3 // L = L / 8
  330. ble dgemm_kernel_L2_M4_40
  331. .align 5
  332. dgemm_kernel_L2_M4_22:
  333. pld [ BO, #B_PRE ]
  334. KERNEL4x2_SUB
  335. KERNEL4x2_SUB
  336. pld [ BO, #B_PRE ]
  337. KERNEL4x2_SUB
  338. KERNEL4x2_SUB
  339. pld [ BO, #B_PRE ]
  340. KERNEL4x2_SUB
  341. KERNEL4x2_SUB
  342. pld [ BO, #B_PRE ]
  343. KERNEL4x2_SUB
  344. KERNEL4x2_SUB
  345. subs L, L, #1
  346. bgt dgemm_kernel_L2_M4_22
  347. dgemm_kernel_L2_M4_40:
  348. ands L , K1, #7 // L = L % 8
  349. ble dgemm_kernel_L2_M4_100
  350. dgemm_kernel_L2_M4_42:
  351. KERNEL4x2_SUB
  352. subs L, L, #1
  353. bgt dgemm_kernel_L2_M4_42
  354. dgemm_kernel_L2_M4_100:
  355. SAVE4x2
  356. dgemm_kernel_L2_M4_END:
  357. subs I, I, #1
  358. bgt dgemm_kernel_L2_M4_20
  359. dgemm_kernel_L2_M2_BEGIN:
  360. ldr I, M
  361. tst I , #3
  362. ble dgemm_kernel_L2_END
  363. tst I, #2 // I = I / 2
  364. ble dgemm_kernel_L2_M1_BEGIN
  365. dgemm_kernel_L2_M2_20:
  366. INIT2x2
  367. mov BO, BC
  368. asrs L , K1, #3 // L = L / 8
  369. ble dgemm_kernel_L2_M2_40
  370. dgemm_kernel_L2_M2_22:
  371. KERNEL2x2_SUB
  372. KERNEL2x2_SUB
  373. KERNEL2x2_SUB
  374. KERNEL2x2_SUB
  375. KERNEL2x2_SUB
  376. KERNEL2x2_SUB
  377. KERNEL2x2_SUB
  378. KERNEL2x2_SUB
  379. subs L, L, #1
  380. bgt dgemm_kernel_L2_M2_22
  381. dgemm_kernel_L2_M2_40:
  382. ands L , K1, #7 // L = L % 8
  383. ble dgemm_kernel_L2_M2_100
  384. dgemm_kernel_L2_M2_42:
  385. KERNEL2x2_SUB
  386. subs L, L, #1
  387. bgt dgemm_kernel_L2_M2_42
  388. dgemm_kernel_L2_M2_100:
  389. SAVE2x2
  390. dgemm_kernel_L2_M2_END:
  391. dgemm_kernel_L2_M1_BEGIN:
  392. tst I, #1 // I = I % 2
  393. ble dgemm_kernel_L2_END
  394. dgemm_kernel_L2_M1_20:
  395. INIT1x2
  396. mov BO, BC
  397. asrs L , K1, #3 // L = L / 8
  398. ble dgemm_kernel_L2_M1_40
  399. dgemm_kernel_L2_M1_22:
  400. KERNEL1x2_SUB
  401. KERNEL1x2_SUB
  402. KERNEL1x2_SUB
  403. KERNEL1x2_SUB
  404. KERNEL1x2_SUB
  405. KERNEL1x2_SUB
  406. KERNEL1x2_SUB
  407. KERNEL1x2_SUB
  408. subs L, L, #1
  409. bgt dgemm_kernel_L2_M1_22
  410. dgemm_kernel_L2_M1_40:
  411. ands L , K1, #7 // L = L % 8
  412. ble dgemm_kernel_L2_M1_100
  413. dgemm_kernel_L2_M1_42:
  414. KERNEL1x2_SUB
  415. subs L, L, #1
  416. bgt dgemm_kernel_L2_M1_42
  417. dgemm_kernel_L2_M1_100:
  418. SAVE1x2
  419. dgemm_kernel_L2_END:
  420. mov r3, BC
  421. mov r4, K1
  422. lsl r4, r4, #4 // k * 2 * 8
  423. add r3, r3, r4 // B = B + K * 2 * 8
  424. mov BC, r3
  425. subs J , #1 // j--
  426. bgt dgemm_kernel_L2_BEGIN
  427. /*********************************************************************************************/
  428. dgemm_kernel_L1_BEGIN:
  429. ldr J , N
  430. tst J , #1
  431. ble dgemm_kernel_L999
  432. ldr CO1, C // CO1 = C
  433. ldr r4 , LDC
  434. add r3 , r4, CO1
  435. str r3 , C // store C
  436. ldr AO, A // AO = A
  437. dgemm_kernel_L1_M4_BEGIN:
  438. ldr I, M
  439. asrs I, I, #2 // I = I / 4
  440. ble dgemm_kernel_L1_M2_BEGIN
  441. dgemm_kernel_L1_M4_20:
  442. INIT4x1
  443. mov BO, BC
  444. asrs L , K1, #3 // L = L / 8
  445. ble dgemm_kernel_L1_M4_40
  446. .align 5
  447. dgemm_kernel_L1_M4_22:
  448. KERNEL4x1_SUB
  449. KERNEL4x1_SUB
  450. KERNEL4x1_SUB
  451. KERNEL4x1_SUB
  452. KERNEL4x1_SUB
  453. KERNEL4x1_SUB
  454. KERNEL4x1_SUB
  455. KERNEL4x1_SUB
  456. subs L, L, #1
  457. bgt dgemm_kernel_L1_M4_22
  458. dgemm_kernel_L1_M4_40:
  459. ands L , K1, #7 // L = L % 8
  460. ble dgemm_kernel_L1_M4_100
  461. dgemm_kernel_L1_M4_42:
  462. KERNEL4x1_SUB
  463. subs L, L, #1
  464. bgt dgemm_kernel_L1_M4_42
  465. dgemm_kernel_L1_M4_100:
  466. SAVE4x1
  467. dgemm_kernel_L1_M4_END:
  468. subs I, I, #1
  469. bgt dgemm_kernel_L1_M4_20
  470. dgemm_kernel_L1_M2_BEGIN:
  471. ldr I, M
  472. tst I , #3
  473. ble dgemm_kernel_L1_END
  474. tst I, #2 // I = I / 2
  475. ble dgemm_kernel_L1_M1_BEGIN
  476. dgemm_kernel_L1_M2_20:
  477. INIT2x1
  478. mov BO, BC
  479. asrs L , K1, #3 // L = L / 8
  480. ble dgemm_kernel_L1_M2_40
  481. dgemm_kernel_L1_M2_22:
  482. KERNEL2x1_SUB
  483. KERNEL2x1_SUB
  484. KERNEL2x1_SUB
  485. KERNEL2x1_SUB
  486. KERNEL2x1_SUB
  487. KERNEL2x1_SUB
  488. KERNEL2x1_SUB
  489. KERNEL2x1_SUB
  490. subs L, L, #1
  491. bgt dgemm_kernel_L1_M2_22
  492. dgemm_kernel_L1_M2_40:
  493. ands L , K1, #7 // L = L % 8
  494. ble dgemm_kernel_L1_M2_100
  495. dgemm_kernel_L1_M2_42:
  496. KERNEL2x1_SUB
  497. subs L, L, #1
  498. bgt dgemm_kernel_L1_M2_42
  499. dgemm_kernel_L1_M2_100:
  500. SAVE2x1
  501. dgemm_kernel_L1_M2_END:
  502. dgemm_kernel_L1_M1_BEGIN:
  503. tst I, #1 // I = I % 2
  504. ble dgemm_kernel_L1_END
  505. dgemm_kernel_L1_M1_20:
  506. INIT1x1
  507. mov BO, BC
  508. asrs L , K1, #3 // L = L / 8
  509. ble dgemm_kernel_L1_M1_40
  510. dgemm_kernel_L1_M1_22:
  511. KERNEL1x1_SUB
  512. KERNEL1x1_SUB
  513. KERNEL1x1_SUB
  514. KERNEL1x1_SUB
  515. KERNEL1x1_SUB
  516. KERNEL1x1_SUB
  517. KERNEL1x1_SUB
  518. KERNEL1x1_SUB
  519. subs L, L, #1
  520. bgt dgemm_kernel_L1_M1_22
  521. dgemm_kernel_L1_M1_40:
  522. ands L , K1, #7 // L = L % 8
  523. ble dgemm_kernel_L1_M1_100
  524. dgemm_kernel_L1_M1_42:
  525. KERNEL1x1_SUB
  526. subs L, L, #1
  527. bgt dgemm_kernel_L1_M1_42
  528. dgemm_kernel_L1_M1_100:
  529. SAVE1x1
  530. dgemm_kernel_L1_END:
  531. dgemm_kernel_L999:
  532. sub r3, fp, #128
  533. vldm r3, { d8 - d15} // restore floating point registers
  534. movs r0, #0 // set return value
  535. sub sp, fp, #24
  536. pop {r4 - r9, fp}
  537. bx lr
  538. EPILOGUE