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.

parameter.c 19 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754
  1. /*********************************************************************/
  2. /* Copyright 2009, 2010 The University of Texas at Austin. */
  3. /* All rights reserved. */
  4. /* */
  5. /* Redistribution and use in source and binary forms, with or */
  6. /* without modification, are permitted provided that the following */
  7. /* conditions are met: */
  8. /* */
  9. /* 1. Redistributions of source code must retain the above */
  10. /* copyright notice, this list of conditions and the following */
  11. /* disclaimer. */
  12. /* */
  13. /* 2. Redistributions in binary form must reproduce the above */
  14. /* copyright notice, this list of conditions and the following */
  15. /* disclaimer in the documentation and/or other materials */
  16. /* provided with the distribution. */
  17. /* */
  18. /* THIS SOFTWARE IS PROVIDED BY THE UNIVERSITY OF TEXAS AT */
  19. /* AUSTIN ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, */
  20. /* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF */
  21. /* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE */
  22. /* DISCLAIMED. IN NO EVENT SHALL THE UNIVERSITY OF TEXAS AT */
  23. /* AUSTIN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, */
  24. /* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES */
  25. /* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE */
  26. /* GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR */
  27. /* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF */
  28. /* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT */
  29. /* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT */
  30. /* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE */
  31. /* POSSIBILITY OF SUCH DAMAGE. */
  32. /* */
  33. /* The views and conclusions contained in the software and */
  34. /* documentation are those of the authors and should not be */
  35. /* interpreted as representing official policies, either expressed */
  36. /* or implied, of The University of Texas at Austin. */
  37. /*********************************************************************/
  38. #include <stdio.h>
  39. #include <string.h>
  40. #include "common.h"
  41. extern int openblas_block_factor();
  42. int get_L2_size(void);
  43. #define DEFAULT_GEMM_P 128
  44. #define DEFAULT_GEMM_Q 128
  45. #define DEFAULT_GEMM_R 128
  46. #define DEFAULT_GEMM_OFFSET_A 0
  47. #define DEFAULT_GEMM_OFFSET_B 0
  48. /* Global Parameter */
  49. #if GEMM_OFFSET_A == gemm_offset_a
  50. BLASLONG gemm_offset_a = DEFAULT_GEMM_OFFSET_A;
  51. #else
  52. BLASLONG gemm_offset_a = GEMM_OFFSET_A;
  53. #endif
  54. #if GEMM_OFFSET_B == gemm_offset_b
  55. BLASLONG gemm_offset_b = DEFAULT_GEMM_OFFSET_B;
  56. #else
  57. BLASLONG gemm_offset_b = GEMM_OFFSET_B;
  58. #endif
  59. #if SHGEMM_P == shgemm_p
  60. BLASLONG shgemm_p = DEFAULT_GEMM_P;
  61. #else
  62. BLASLONG shgemm_p = SHGEMM_P;
  63. #endif
  64. #if SGEMM_P == sgemm_p
  65. BLASLONG sgemm_p = DEFAULT_GEMM_P;
  66. #else
  67. BLASLONG sgemm_p = SGEMM_P;
  68. #endif
  69. #if DGEMM_P == dgemm_p
  70. BLASLONG dgemm_p = DEFAULT_GEMM_P;
  71. #else
  72. BLASLONG dgemm_p = DGEMM_P;
  73. #endif
  74. #if CGEMM_P == cgemm_p
  75. BLASLONG cgemm_p = DEFAULT_GEMM_P;
  76. #else
  77. BLASLONG cgemm_p = CGEMM_P;
  78. #endif
  79. #if ZGEMM_P == zgemm_p
  80. BLASLONG zgemm_p = DEFAULT_GEMM_P;
  81. #else
  82. BLASLONG zgemm_p = ZGEMM_P;
  83. #endif
  84. #if SHGEMM_Q == shgemm_q
  85. BLASLONG shgemm_q = DEFAULT_GEMM_Q;
  86. #else
  87. BLASLONG shgemm_q = SHGEMM_Q;
  88. #endif
  89. #if SGEMM_Q == sgemm_q
  90. BLASLONG sgemm_q = DEFAULT_GEMM_Q;
  91. #else
  92. BLASLONG sgemm_q = SGEMM_Q;
  93. #endif
  94. #if DGEMM_Q == dgemm_q
  95. BLASLONG dgemm_q = DEFAULT_GEMM_Q;
  96. #else
  97. BLASLONG dgemm_q = DGEMM_Q;
  98. #endif
  99. #if CGEMM_Q == cgemm_q
  100. BLASLONG cgemm_q = DEFAULT_GEMM_Q;
  101. #else
  102. BLASLONG cgemm_q = CGEMM_Q;
  103. #endif
  104. #if ZGEMM_Q == zgemm_q
  105. BLASLONG zgemm_q = DEFAULT_GEMM_Q;
  106. #else
  107. BLASLONG zgemm_q = ZGEMM_Q;
  108. #endif
  109. #if SHGEMM_R == shgemm_r
  110. BLASLONG shgemm_r = DEFAULT_GEMM_R;
  111. #else
  112. BLASLONG shgemm_r = SHGEMM_R;
  113. #endif
  114. #if SGEMM_R == sgemm_r
  115. BLASLONG sgemm_r = DEFAULT_GEMM_R;
  116. #else
  117. BLASLONG sgemm_r = SGEMM_R;
  118. #endif
  119. #if DGEMM_R == dgemm_r
  120. BLASLONG dgemm_r = DEFAULT_GEMM_R;
  121. #else
  122. BLASLONG dgemm_r = DGEMM_R;
  123. #endif
  124. #if CGEMM_R == cgemm_r
  125. BLASLONG cgemm_r = DEFAULT_GEMM_R;
  126. #else
  127. BLASLONG cgemm_r = CGEMM_R;
  128. #endif
  129. #if ZGEMM_R == zgemm_r
  130. BLASLONG zgemm_r = DEFAULT_GEMM_R;
  131. #else
  132. BLASLONG zgemm_r = ZGEMM_R;
  133. #endif
  134. #if defined(EXPRECISION) || defined(QUAD_PRECISION)
  135. #if QGEMM_P == qgemm_p
  136. BLASLONG qgemm_p = DEFAULT_GEMM_P;
  137. #else
  138. BLASLONG qgemm_p = QGEMM_P;
  139. #endif
  140. #if XGEMM_P == xgemm_p
  141. BLASLONG xgemm_p = DEFAULT_GEMM_P;
  142. #else
  143. BLASLONG xgemm_p = XGEMM_P;
  144. #endif
  145. #if QGEMM_Q == qgemm_q
  146. BLASLONG qgemm_q = DEFAULT_GEMM_Q;
  147. #else
  148. BLASLONG qgemm_q = QGEMM_Q;
  149. #endif
  150. #if XGEMM_Q == xgemm_q
  151. BLASLONG xgemm_q = DEFAULT_GEMM_Q;
  152. #else
  153. BLASLONG xgemm_q = XGEMM_Q;
  154. #endif
  155. #if QGEMM_R == qgemm_r
  156. BLASLONG qgemm_r = DEFAULT_GEMM_R;
  157. #else
  158. BLASLONG qgemm_r = QGEMM_R;
  159. #endif
  160. #if XGEMM_R == xgemm_r
  161. BLASLONG xgemm_r = DEFAULT_GEMM_R;
  162. #else
  163. BLASLONG xgemm_r = XGEMM_R;
  164. #endif
  165. #endif
  166. #if defined(ARCH_X86) || defined(ARCH_X86_64)
  167. int get_L2_size(void){
  168. int eax, ebx, ecx, edx;
  169. #if defined(ATHLON) || defined(OPTERON) || defined(BARCELONA) || defined(BOBCAT) || defined(BULLDOZER) || \
  170. defined(CORE_PRESCOTT) || defined(CORE_CORE2) || defined(PENRYN) || defined(DUNNINGTON) || \
  171. defined(CORE_NEHALEM) || defined(CORE_SANDYBRIDGE) || defined(ATOM) || defined(GENERIC) || \
  172. defined(PILEDRIVER) || defined(HASWELL) || defined(STEAMROLLER) || defined(EXCAVATOR) || defined(ZEN) || defined(SKYLAKEX)
  173. cpuid(0x80000006, &eax, &ebx, &ecx, &edx);
  174. return BITMASK(ecx, 16, 0xffff);
  175. #else
  176. int info[15];
  177. int i;
  178. cpuid(2, &eax, &ebx, &ecx, &edx);
  179. info[ 0] = BITMASK(eax, 8, 0xff);
  180. info[ 1] = BITMASK(eax, 16, 0xff);
  181. info[ 2] = BITMASK(eax, 24, 0xff);
  182. info[ 3] = BITMASK(ebx, 0, 0xff);
  183. info[ 4] = BITMASK(ebx, 8, 0xff);
  184. info[ 5] = BITMASK(ebx, 16, 0xff);
  185. info[ 6] = BITMASK(ebx, 24, 0xff);
  186. info[ 7] = BITMASK(ecx, 0, 0xff);
  187. info[ 8] = BITMASK(ecx, 8, 0xff);
  188. info[ 9] = BITMASK(ecx, 16, 0xff);
  189. info[10] = BITMASK(ecx, 24, 0xff);
  190. info[11] = BITMASK(edx, 0, 0xff);
  191. info[12] = BITMASK(edx, 8, 0xff);
  192. info[13] = BITMASK(edx, 16, 0xff);
  193. info[14] = BITMASK(edx, 24, 0xff);
  194. for (i = 0; i < 15; i++){
  195. switch (info[i]){
  196. case 0x3b :
  197. case 0x41 :
  198. case 0x79 :
  199. return 128;
  200. break;
  201. case 0x3c :
  202. case 0x42 :
  203. case 0x7a :
  204. case 0x7e :
  205. case 0x82 :
  206. return 256;
  207. break;
  208. case 0x43 :
  209. case 0x7b :
  210. case 0x7f :
  211. case 0x83 :
  212. case 0x86 :
  213. return 512;
  214. break;
  215. case 0x44 :
  216. case 0x78 :
  217. case 0x7c :
  218. case 0x84 :
  219. case 0x87 :
  220. return 1024;
  221. break;
  222. case 0x45 :
  223. case 0x7d :
  224. case 0x85 :
  225. return 2048;
  226. case 0x49 :
  227. return 4096;
  228. break;
  229. }
  230. }
  231. /* Never reached */
  232. return 0;
  233. #endif
  234. }
  235. void blas_set_parameter(void){
  236. int factor;
  237. #if defined(BULLDOZER) || defined(PILEDRIVER) || defined(SANDYBRIDGE) || defined(NEHALEM) || defined(HASWELL) || defined(STEAMROLLER) || defined(EXCAVATOR) || defined(ZEN) || defined(SKYLAKEX)
  238. int size = 16;
  239. #else
  240. int size = get_L2_size();
  241. #endif
  242. #if defined(CORE_KATMAI) || defined(CORE_COPPERMINE) || defined(CORE_BANIAS)
  243. size >>= 7;
  244. #if defined(CORE_BANIAS) && (HAVE_HIT > 1)
  245. sgemm_p = 64 / HAVE_HIT * size;
  246. dgemm_p = 32 / HAVE_HIT * size;
  247. cgemm_p = 32 / HAVE_HIT * size;
  248. zgemm_p = 16 / HAVE_HIT * size;
  249. #ifdef EXPRECISION
  250. qgemm_p = 16 / HAVE_HIT * size;
  251. xgemm_p = 8 / HAVE_HIT * size;
  252. #endif
  253. #ifdef QUAD_PRECISION
  254. qgemm_p = 8 / HAVE_HIT * size;
  255. xgemm_p = 4 / HAVE_HIT * size;
  256. #endif
  257. #else
  258. sgemm_p = 64 * size;
  259. dgemm_p = 32 * size;
  260. cgemm_p = 32 * size;
  261. zgemm_p = 16 * size;
  262. #ifdef EXPRECISION
  263. qgemm_p = 16 * size;
  264. xgemm_p = 8 * size;
  265. #endif
  266. #ifdef QUAD_PRECISION
  267. qgemm_p = 8 * size;
  268. xgemm_p = 4 * size;
  269. #endif
  270. #endif
  271. #endif
  272. #if defined(CORE_NORTHWOOD)
  273. size >>= 7;
  274. #ifdef ALLOC_HUGETLB
  275. sgemm_p = 128 * size;
  276. dgemm_p = 64 * size;
  277. cgemm_p = 64 * size;
  278. zgemm_p = 32 * size;
  279. #ifdef EXPRECISION
  280. qgemm_p = 32 * size;
  281. xgemm_p = 16 * size;
  282. #endif
  283. #ifdef QUAD_PRECISION
  284. qgemm_p = 16 * size;
  285. xgemm_p = 8 * size;
  286. #endif
  287. #else
  288. sgemm_p = 96 * size;
  289. dgemm_p = 48 * size;
  290. cgemm_p = 48 * size;
  291. zgemm_p = 24 * size;
  292. #ifdef EXPRECISION
  293. qgemm_p = 24 * size;
  294. xgemm_p = 12 * size;
  295. #endif
  296. #ifdef QUAD_PRECISION
  297. qgemm_p = 12 * size;
  298. xgemm_p = 6 * size;
  299. #endif
  300. #endif
  301. #endif
  302. #if defined(CORE_CORE2)
  303. size >>= 9;
  304. sgemm_p = 92 * size;
  305. dgemm_p = 46 * size;
  306. cgemm_p = 46 * size;
  307. zgemm_p = 23 * size;
  308. #ifdef EXPRECISION
  309. qgemm_p = 23 * size;
  310. xgemm_p = 11 * size;
  311. #endif
  312. #ifdef QUAD_PRECISION
  313. qgemm_p = 11 * size;
  314. xgemm_p = 5 * size;
  315. #endif
  316. #endif
  317. #if defined(PENRYN)
  318. size >>= 9;
  319. sgemm_p = 1024;
  320. dgemm_p = 512;
  321. cgemm_p = 512;
  322. zgemm_p = 256;
  323. #ifdef EXPRECISION
  324. qgemm_p = 256;
  325. xgemm_p = 128;
  326. #endif
  327. #ifdef QUAD_PRECISION
  328. qgemm_p = 21 * size + 4;
  329. xgemm_p = 10 * size + 2;
  330. #endif
  331. #endif
  332. #if defined(DUNNINGTON)
  333. size >>= 9;
  334. sgemm_p = 384;
  335. dgemm_p = 384;
  336. cgemm_p = 384;
  337. zgemm_p = 384;
  338. #ifdef EXPRECISION
  339. qgemm_p = 384;
  340. xgemm_p = 384;
  341. #endif
  342. #ifdef QUAD_PRECISION
  343. qgemm_p = 21 * size + 4;
  344. xgemm_p = 10 * size + 2;
  345. #endif
  346. #endif
  347. #if defined(NEHALEM)
  348. sgemm_p = 1024;
  349. dgemm_p = 512;
  350. cgemm_p = 512;
  351. zgemm_p = 256;
  352. #ifdef EXPRECISION
  353. qgemm_p = 256;
  354. xgemm_p = 128;
  355. #endif
  356. #endif
  357. #if defined(SANDYBRIDGE)
  358. sgemm_p = 1024;
  359. dgemm_p = 512;
  360. cgemm_p = 512;
  361. zgemm_p = 256;
  362. #ifdef EXPRECISION
  363. qgemm_p = 256;
  364. xgemm_p = 128;
  365. #endif
  366. #endif
  367. #if defined(CORE_PRESCOTT) || defined(GENERIC)
  368. size >>= 6;
  369. if (size > 16) size = 16;
  370. sgemm_p = 56 * size;
  371. dgemm_p = 28 * size;
  372. cgemm_p = 28 * size;
  373. zgemm_p = 14 * size;
  374. #ifdef EXPRECISION
  375. qgemm_p = 14 * size;
  376. xgemm_p = 7 * size;
  377. #endif
  378. #ifdef QUAD_PRECISION
  379. qgemm_p = 7 * size;
  380. xgemm_p = 3 * size;
  381. #endif
  382. #endif
  383. #if defined(CORE_OPTERON)
  384. sgemm_p = 224 + 14 * (size >> 5);
  385. dgemm_p = 112 + 14 * (size >> 6);
  386. cgemm_p = 116 + 14 * (size >> 6);
  387. zgemm_p = 58 + 14 * (size >> 7);
  388. #ifdef EXPRECISION
  389. qgemm_p = 58 + 14 * (size >> 7);
  390. xgemm_p = 29 + 14 * (size >> 8);
  391. #endif
  392. #ifdef QUAD_PRECISION
  393. qgemm_p = 29 + 14 * (size >> 8);
  394. xgemm_p = 15 + 14 * (size >> 9);
  395. #endif
  396. #endif
  397. #if defined(ATOM)
  398. size >>= 8;
  399. sgemm_p = 256;
  400. dgemm_p = 128;
  401. cgemm_p = 128;
  402. zgemm_p = 64;
  403. #ifdef EXPRECISION
  404. qgemm_p = 64;
  405. xgemm_p = 32;
  406. #endif
  407. #ifdef QUAD_PRECISION
  408. qgemm_p = 32;
  409. xgemm_p = 16;
  410. #endif
  411. #endif
  412. #if defined(CORE_BARCELONA) || defined(CORE_BOBCAT)
  413. size >>= 8;
  414. sgemm_p = 232 * size;
  415. dgemm_p = 116 * size;
  416. cgemm_p = 116 * size;
  417. zgemm_p = 58 * size;
  418. #ifdef EXPRECISION
  419. qgemm_p = 58 * size;
  420. xgemm_p = 26 * size;
  421. #endif
  422. #ifdef QUAD_PRECISION
  423. qgemm_p = 26 * size;
  424. xgemm_p = 13 * size;
  425. #endif
  426. #endif
  427. factor=openblas_block_factor();
  428. if (factor>0) {
  429. if (factor < 10) factor = 10;
  430. if (factor > 200) factor = 200;
  431. sgemm_p = ((long)((double)sgemm_p * (double)factor * 1.e-2)) & ~7L;
  432. dgemm_p = ((long)((double)dgemm_p * (double)factor * 1.e-2)) & ~7L;
  433. cgemm_p = ((long)((double)cgemm_p * (double)factor * 1.e-2)) & ~7L;
  434. zgemm_p = ((long)((double)zgemm_p * (double)factor * 1.e-2)) & ~7L;
  435. #ifdef EXPRECISION
  436. qgemm_p = ((long)((double)qgemm_p * (double)factor * 1.e-2)) & ~7L;
  437. xgemm_p = ((long)((double)xgemm_p * (double)factor * 1.e-2)) & ~7L;
  438. #endif
  439. }
  440. if (sgemm_p == 0) sgemm_p = 64;
  441. if (dgemm_p == 0) dgemm_p = 64;
  442. if (cgemm_p == 0) cgemm_p = 64;
  443. if (zgemm_p == 0) zgemm_p = 64;
  444. #ifdef EXPRECISION
  445. if (qgemm_p == 0) qgemm_p = 64;
  446. if (xgemm_p == 0) xgemm_p = 64;
  447. #endif
  448. #ifdef QUAD_PRECISION
  449. if (qgemm_p == 0) qgemm_p = 64;
  450. if (xgemm_p == 0) xgemm_p = 64;
  451. #endif
  452. sgemm_p = ((sgemm_p + SGEMM_UNROLL_M - 1)/SGEMM_UNROLL_M) * SGEMM_UNROLL_M;
  453. dgemm_p = ((dgemm_p + DGEMM_UNROLL_M - 1)/DGEMM_UNROLL_M) * DGEMM_UNROLL_M;
  454. cgemm_p = ((cgemm_p + CGEMM_UNROLL_M - 1)/CGEMM_UNROLL_M) * CGEMM_UNROLL_M;
  455. zgemm_p = ((zgemm_p + ZGEMM_UNROLL_M - 1)/ZGEMM_UNROLL_M) * ZGEMM_UNROLL_M;
  456. #ifdef QUAD_PRECISION
  457. qgemm_p = ((qgemm_p + QGEMM_UNROLL_M - 1)/QGEMM_UNROLL_M) * QGEMM_UNROLL_M;
  458. xgemm_p = ((xgemm_p + XGEMM_UNROLL_M - 1)/XGEMM_UNROLL_M) * XGEMM_UNROLL_M;
  459. #endif
  460. sgemm_r = (((BUFFER_SIZE - ((SGEMM_P * SGEMM_Q * 4 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (SGEMM_Q * 4)) - 15) & ~15;
  461. dgemm_r = (((BUFFER_SIZE - ((DGEMM_P * DGEMM_Q * 8 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (DGEMM_Q * 8)) - 15) & ~15;
  462. cgemm_r = (((BUFFER_SIZE - ((CGEMM_P * CGEMM_Q * 8 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (CGEMM_Q * 8)) - 15) & ~15;
  463. zgemm_r = (((BUFFER_SIZE - ((ZGEMM_P * ZGEMM_Q * 16 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (ZGEMM_Q * 16)) - 15) & ~15;
  464. #if defined(EXPRECISION) || defined(QUAD_PRECISION)
  465. qgemm_r = (((BUFFER_SIZE - ((QGEMM_P * QGEMM_Q * 16 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (QGEMM_Q * 16)) - 15) & ~15;
  466. xgemm_r = (((BUFFER_SIZE - ((XGEMM_P * XGEMM_Q * 32 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (XGEMM_Q * 32)) - 15) & ~15;
  467. #endif
  468. #if 0
  469. fprintf(stderr, "SGEMM ... %3d, %3d, %3d\n", SGEMM_P, SGEMM_Q, SGEMM_R);
  470. fprintf(stderr, "DGEMM ... %3d, %3d, %3d\n", DGEMM_P, DGEMM_Q, DGEMM_R);
  471. fprintf(stderr, "CGEMM ... %3d, %3d, %3d\n", CGEMM_P, CGEMM_Q, CGEMM_R);
  472. fprintf(stderr, "ZGEMM ... %3d, %3d, %3d\n", ZGEMM_P, ZGEMM_Q, ZGEMM_R);
  473. #endif
  474. return;
  475. }
  476. #if 0
  477. int get_current_cpu_info(void){
  478. int nlprocs, ncores, cmplegacy;
  479. int htt = 0;
  480. int apicid = 0;
  481. #if defined(CORE_PRESCOTT) || defined(CORE_OPTERON)
  482. int eax, ebx, ecx, edx;
  483. cpuid(1, &eax, &ebx, &ecx, &edx);
  484. nlprocs = BITMASK(ebx, 16, 0xff);
  485. apicid = BITMASK(ebx, 24, 0xff);
  486. htt = BITMASK(edx, 28, 0x01);
  487. #endif
  488. #if defined(CORE_PRESCOTT)
  489. cpuid(4, &eax, &ebx, &ecx, &edx);
  490. ncores = BITMASK(eax, 26, 0x3f);
  491. if (htt == 0) nlprocs = 0;
  492. #endif
  493. #if defined(CORE_OPTERON)
  494. cpuid(0x80000008, &eax, &ebx, &ecx, &edx);
  495. ncores = BITMASK(ecx, 0, 0xff);
  496. cpuid(0x80000001, &eax, &ebx, &ecx, &edx);
  497. cmplegacy = BITMASK(ecx, 1, 0x01);
  498. if (htt == 0) {
  499. nlprocs = 0;
  500. ncores = 0;
  501. cmplegacy = 0;
  502. }
  503. #endif
  504. ncores ++;
  505. fprintf(stderr, "APICID = %d Number of core = %d\n", apicid, ncores);
  506. return 0;
  507. }
  508. #endif
  509. #endif
  510. #if defined(ARCH_IA64)
  511. static inline BLASULONG cpuid(BLASULONG regnum){
  512. BLASULONG value;
  513. #ifndef __ECC
  514. asm ("mov %0=cpuid[%r1]" : "=r"(value) : "rO"(regnum));
  515. #else
  516. value = __getIndReg(_IA64_REG_INDR_CPUID, regnum);
  517. #endif
  518. return value;
  519. }
  520. #if 1
  521. void blas_set_parameter(void){
  522. BLASULONG cpuid3, size;
  523. cpuid3 = cpuid(3);
  524. size = BITMASK(cpuid3, 16, 0xff);
  525. shgemm_p = 192 * (size + 1);
  526. sgemm_p = 192 * (size + 1);
  527. dgemm_p = 96 * (size + 1);
  528. cgemm_p = 96 * (size + 1);
  529. zgemm_p = 48 * (size + 1);
  530. #ifdef EXPRECISION
  531. qgemm_p = 64 * (size + 1);
  532. xgemm_p = 32 * (size + 1);
  533. #endif
  534. #ifdef QUAD_PRECISION
  535. qgemm_p = 32 * (size + 1);
  536. xgemm_p = 16 * (size + 1);
  537. #endif
  538. shgemm_r = (((BUFFER_SIZE - ((SHGEMM_P * SHGEMM_Q * 4 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (SHGEMM_Q * 4)) - 15) & ~15;
  539. sgemm_r = (((BUFFER_SIZE - ((SGEMM_P * SGEMM_Q * 4 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (SGEMM_Q * 4)) - 15) & ~15;
  540. dgemm_r = (((BUFFER_SIZE - ((DGEMM_P * DGEMM_Q * 8 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (DGEMM_Q * 8)) - 15) & ~15;
  541. cgemm_r = (((BUFFER_SIZE - ((CGEMM_P * CGEMM_Q * 8 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (CGEMM_Q * 8)) - 15) & ~15;
  542. zgemm_r = (((BUFFER_SIZE - ((ZGEMM_P * ZGEMM_Q * 16 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (ZGEMM_Q * 16)) - 15) & ~15;
  543. #if defined(EXPRECISION) || defined(QUAD_PRECISION)
  544. qgemm_r = (((BUFFER_SIZE - ((QGEMM_P * QGEMM_Q * 16 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (QGEMM_Q * 16)) - 15) & ~15;
  545. xgemm_r = (((BUFFER_SIZE - ((XGEMM_P * XGEMM_Q * 32 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (XGEMM_Q * 32)) - 15) & ~15;
  546. #endif
  547. return;
  548. }
  549. #else
  550. #define IA64_SYS_NAME "/sys/devices/system/cpu/cpu0/cache/index3/size"
  551. #define IA64_PROC_NAME "/proc/pal/cpu0/cache_info"
  552. void blas_set_parameter(void){
  553. BLASULONG cpuid3;
  554. int size = 0;
  555. #if 1
  556. char buffer[128];
  557. FILE *infile;
  558. if ((infile = fopen(IA64_SYS_NAME, "r")) != NULL) {
  559. fgets(buffer, sizeof(buffer), infile);
  560. fclose(infile);
  561. size = atoi(buffer) / 1536;
  562. }
  563. if (size <= 0) {
  564. if ((infile = fopen(IA64_PROC_NAME, "r")) != NULL) {
  565. while(fgets(buffer, sizeof(buffer), infile) != NULL) {
  566. if ((!strncmp("Data/Instruction Cache level 3", buffer, 30))) break;
  567. }
  568. fgets(buffer, sizeof(buffer), infile);
  569. fclose(infile);
  570. *strstr(buffer, "bytes") = (char)NULL;
  571. size = atoi(strchr(buffer, ':') + 1) / 1572864;
  572. }
  573. }
  574. #endif
  575. /* The last resort */
  576. if (size <= 0) {
  577. cpuid3 = cpuid(3);
  578. size = BITMASK(cpuid3, 16, 0xff) + 1;
  579. }
  580. sgemm_p = 320 * size;
  581. dgemm_p = 160 * size;
  582. cgemm_p = 160 * size;
  583. zgemm_p = 80 * size;
  584. #ifdef EXPRECISION
  585. qgemm_p = 80 * size;
  586. xgemm_p = 40 * size;
  587. #endif
  588. sgemm_r = (((BUFFER_SIZE - ((SGEMM_P * SGEMM_Q * 4 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (SGEMM_Q * 4)) - 15) & ~15;
  589. dgemm_r = (((BUFFER_SIZE - ((DGEMM_P * DGEMM_Q * 8 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (DGEMM_Q * 8)) - 15) & ~15;
  590. cgemm_r = (((BUFFER_SIZE - ((CGEMM_P * CGEMM_Q * 8 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (CGEMM_Q * 8)) - 15) & ~15;
  591. zgemm_r = (((BUFFER_SIZE - ((ZGEMM_P * ZGEMM_Q * 16 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (ZGEMM_Q * 16)) - 15) & ~15;
  592. #ifdef EXPRECISION
  593. qgemm_r = (((BUFFER_SIZE - ((QGEMM_P * QGEMM_Q * 16 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (QGEMM_Q * 16)) - 15) & ~15;
  594. xgemm_r = (((BUFFER_SIZE - ((XGEMM_P * XGEMM_Q * 32 + GEMM_OFFSET_A + GEMM_ALIGN) & ~GEMM_ALIGN)) / (XGEMM_Q * 32)) - 15) & ~15;
  595. #endif
  596. return;
  597. }
  598. #endif
  599. #endif
  600. #if defined(ARCH_MIPS64)
  601. void blas_set_parameter(void){
  602. #if defined(LOONGSON3A)
  603. #ifdef SMP
  604. if(blas_num_threads == 1){
  605. #endif
  606. //single thread
  607. dgemm_r = 1024;
  608. #ifdef SMP
  609. }else{
  610. //multi thread
  611. dgemm_r = 200;
  612. }
  613. #endif
  614. #endif
  615. #if defined(LOONGSON3B)
  616. #ifdef SMP
  617. if(blas_num_threads == 1 || blas_num_threads == 2){
  618. #endif
  619. //single thread
  620. dgemm_r = 640;
  621. #ifdef SMP
  622. }else{
  623. //multi thread
  624. dgemm_r = 160;
  625. }
  626. #endif
  627. #endif
  628. }
  629. #endif
  630. #if defined(ARCH_ARM64)
  631. void blas_set_parameter(void)
  632. {
  633. }
  634. #endif