-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathcountsse2_amd64.s
636 lines (566 loc) · 15.2 KB
/
countsse2_amd64.s
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
#include "textflag.h"
// An SSE2 based kernel first doing a 15-fold CSA reduction and then
// a 16-fold CSA reduction, carrying over place-value vectors between
// iterations. Required CPU extension: SSE2.
// magic transposition constants
DATA magic<> +0(SB)/8, $0x8040201008040201
DATA magic<>+ 8(SB)/8, $0xaaaaaaaa55555555
DATA magic<>+16(SB)/8, $0xcccccccc33333333
DATA magic<>+24(SB)/8, $0x00ff00ff0f0f0f0f
GLOBL magic<>(SB), RODATA|NOPTR, $32
// sliding window for head/tail loads. Unfortunately, there doesn't
// seem to be a good way to do this with less memory wasted.
DATA window<> +0(SB)/8, $0x0000000000000000
DATA window<> +8(SB)/8, $0x0000000000000000
DATA window<>+16(SB)/8, $0xffffffffffffffff
DATA window<>+24(SB)/8, $0xffffffffffffffff
GLOBL window<>(SB), RODATA|NOPTR, $32
// B:A = A+B+C
#define CSA(A, B, C) \
PXOR C, B \
PXOR A, C \
PXOR B, A \
POR C, B \
PXOR A, B
// Process 4 bytes from X4. Add low word counts to L, high to H
// assumes mask loaded into X2. Trashes X4, X5.
#define COUNT4(L, H) \ // X4 = ----:----:----:3210
PUNPCKLBW X4, X4 \ // X4 = ----:----:3322:1100
PUNPCKLWL X4, X4 \ // X4 = 3333:2222:1111:0000
PSHUFD $0xfa, X4, X5 \ // X5 = 3333:3333:2222:2222
PUNPCKLLQ X4, X4 \ // X5 = 1111:1111:0000:0000
PAND X6, X4 \
PAND X6, X5 \
PCMPEQB X6, X4 \
PCMPEQB X6, X5 \
PSUBB X4, L \
PSUBB X5, H
// zero extend X from bytes into words and add to the counter vectors
// S1 and S2. X7 is expected to be a zero register, X6 and X are trashed.
#define ACCUM(S1, S2, X) \
MOVOA X, X6 \
PUNPCKLBW X7, X \
PUNPCKHBW X7, X6 \
PADDW X, S1 \
PADDW X6, S2
// Generic kernel. This function expects a pointer to a width-specific
// accumulation funciton in BX, a possibly unaligned input buffer in SI,
// counters in DI and a remaining length in CX.
TEXT countsse2<>(SB), NOSPLIT, $32-0
// constants for processing the head
MOVQ magic<>+0(SB), X6 // bit position mask
PSHUFD $0x44, X6, X6 // broadcast into both qwords
PXOR X7, X7 // zero register
PXOR X8, X8 // counter registers
PXOR X10, X10
PXOR X12, X12
PXOR X14, X14
CMPQ CX, $15*16 // is the CSA kernel worth using?
JLT runt
// load head until alignment/end is reached
MOVL SI, DX
ANDL $15, DX // offset of the buffer start from 16 byte alignment
MOVL $16, AX
SUBL DX, AX // number of bytes til alignment is reached (head length)
SUBQ DX, SI // align source to 16 bytes
ADDQ DX, CX // and account for head length
MOVQ $window<>(SB), DX // load window mask base pointer
MOVOU (DX)(AX*1), X2 // load mask of the bytes that are part of the head
PAND (SI), X2 // load head and mask out bytes that are not in the head
// load 240 - 16 bytes from buf and sum them into X3:X2:X1:X0
MOVOA 1*16(SI), X1
MOVOA 2*16(SI), X0
MOVOA 3*16(SI), X5
MOVOA 4*16(SI), X4
MOVOA 5*16(SI), X3
CSA(X0, X1, X2)
MOVOA 6*16(SI), X7
MOVOA 7*16(SI), X6
MOVOA 8*16(SI), X2
CSA(X3, X4, X5)
MOVOA 9*16(SI), X5
CSA(X2, X6, X7)
MOVOA 10*16(SI), X7
CSA(X0, X5, X3)
MOVOA 11*16(SI), X3
CSA(X1, X4, X6)
MOVOA 12*16(SI), X6
CSA(X0, X2, X7)
MOVOA 13*16(SI), X7
PXOR X9, X9 // initialise remaining counters
PXOR X11, X11
CSA(X3, X7, X6)
MOVOA 14*16(SI), X6
CSA(X1, X2, X5)
ADDQ $15*16, SI
CSA(X0, X3, X6)
MOVL $65535, AX // space left til overflow could occur in Y8--Y11
CSA(X1, X3, X7)
PXOR X13, X13
PXOR X15, X15
CSA(X2, X3, X4)
SUBQ $(15+16)*16, CX // enough data left to process?
JLT post
// load 256 bytes from buf, add them to X0..X3 into X0..X4
vec: MOVOA 0*16(SI), X4
MOVOA 1*16(SI), X5
MOVOU X8, X8save-32(SP) // stash some counters to give us
MOVOU X9, X9save-16(SP) // more registers to play with
MOVOA 2*16(SI), X6
MOVOA 3*16(SI), X7
MOVOA 4*16(SI), X8
MOVOA 5*16(SI), X9
CSA(X0, X5, X4)
MOVOA 6*16(SI), X4
CSA(X6, X8, X7)
MOVOA 7*16(SI), X7
CSA(X1, X8, X5)
MOVOA 8*16(SI), X5
CSA(X0, X6, X9)
MOVOA 9*16(SI), X9
CSA(X4, X5, X7)
MOVOA 10*16(SI), X7
CSA(X1, X5, X6)
MOVOA 11*16(SI), X6
CSA(X0, X4, X9)
MOVOA 12*16(SI), X9
CSA(X2, X5, X8)
MOVOA 13*16(SI), X8
CSA(X0, X6, X7)
MOVOA 14*16(SI), X7
CSA(X1, X4, X6)
MOVOA 15*16(SI), X6
CSA(X7, X8, X9)
MOVOU magic<>+8(SB), X9 // 55555555, aaaaaaaa, 33333333, cccccccc
CSA(X0, X6, X7)
ADDQ $16*16, SI
#define D 90
PREFETCHT0 (D+ 0)*16(SI)
CSA(X1, X6, X8)
PREFETCHT0 (D+ 4)*16(SI)
CSA(X2, X4, X6)
PREFETCHT0 (D+ 8)*16(SI)
CSA(X3, X4, X5)
PREFETCHT0 (D+12)*16(SI)
MOVQ magic<>+24(SB), X8 // 0f0f0f0f, 00ff00ff
// now X0..X4 hold counters; preserve X0..X4 for the next round
// and add X4 to the the counters.
// split into even/odd and reduce into crumbs
PSHUFD $0x00, X9, X7 // X7 = 55..55
MOVOA X4, X5
PAND X7, X5 // X5 = 02468ace x8
PANDN X4, X7 // X7 = 13579bdf x8
PSRLL $1, X7
MOVOA X5, X4
PUNPCKLQDQ X7, X4
PUNPCKHQDQ X7, X5
PADDL X5, X4 // X4 = 02468ace x4 13579bdf x4
// split again into nibbles
PSHUFD $0xaa, X9, X5 // X7 = 33..33
MOVOA X5, X7
PANDN X4, X5 // X5 = 26ae x4 37bf x4
PAND X7, X4 // X4 = 048c x4 159d x4
PSRLL $2, X5
// split into bytes and shuffle into order
PSHUFD $0x00, X8, X6 // X6 = 0f..0f
MOVOA X6, X7
PANDN X4, X6 // X6 = 4c x4 5d x4
PAND X7, X4 // X4 = 08 x4 19 x4
MOVOA X7, X9
PANDN X5, X7 // X7 = 6e x4 7f x4
PAND X9, X5 // X5 = 2a x4 3b x4
PSLLL $4, X4
PSLLL $4, X5
MOVOA X4, X9
PUNPCKLWL X5, X4 // X4 = 082a x4
PUNPCKHWL X5, X9 // X9 = 193b x4
MOVOA X6, X5
PUNPCKLWL X7, X5 // X5 = 4c6e x4
PUNPCKHWL X7, X6 // X6 = 5d7f x4
MOVOA X4, X7
PUNPCKLWL X9, X4 // X4 = 08192a3b[0:1]
PUNPCKHWL X9, X7 // X7 = 08192a3b[2:3]
MOVOA X5, X9
PUNPCKLWL X6, X5 // X5 = 4c5d6e7f[0:1]
PUNPCKHWL X6, X9 // X9 = 4c5d6e7f[2:3]
MOVOA X4, X6
PUNPCKLQDQ X5, X4 // X4 = 08192a3b4c5d6e7f[0]
PUNPCKHQDQ X5, X6 // X6 = 08192a3b4c5d6e7f[1]
MOVOA X7, X5
PUNPCKLQDQ X9, X5 // X5 = 08192a3b4c5d6e7f[2]
PUNPCKHQDQ X9, X7 // X7 = 08192a3b4c5d6e7f[3]
// split into words and add to counters
PSHUFD $0x55, X8, X8 // X8 = 00ff..00ff
MOVOA X6, X9
PAND X8, X6 // X6 = 01234678[1]
PSRLW $8, X9 // X9 = 89abcdef[1]
PADDW X6, X10
PADDW X9, X11
MOVOA X8, X6
MOVOU X8save-32(SP), X8
MOVOA X5, X9
PAND X6, X5 // X5 = 01234567[2]
PSRLW $8, X9 // X9 = 89abcdef[2]
PADDW X5, X12
PADDW X9, X13
MOVOU X9save-16(SP), X9
MOVOA X7, X5
PAND X6, X7 // X7 = 01234567[3]
PSRLW $8, X5 // X5 = 89abcdef[3]
PADDW X7, X14
PADDW X5, X15
MOVOA X4, X5
PAND X6, X4 // X4 = 01234567[0]
PSRLW $8, X5 // X5 = 89abcdef[0]
PADDW X4, X8
PADDW X5, X9
SUBL $16*2, AX // account for possible overflow
CMPL AX, $(15+15)*2 // enough space left in the counters?
JGE have_space
PXOR X7, X7
CALL *BX // call accumulation function
PXOR X8, X8 // clear counters for next round
PXOR X9, X9
PXOR X10, X10
PXOR X11, X11
PXOR X12, X12
PXOR X13, X13
PXOR X14, X14
PXOR X15, X15
MOVL $65535, AX // space left til overflow could occur
have_space:
SUBQ $16*16, CX // account for bytes consumed
JGE vec
post: MOVQ magic<>+8(SB), X5 // load magic constants
PSHUFD $0x55, X5, X6 // 0xaaaaaaaa
PSHUFD $0x00, X5, X7 // 0x55555555
// group X0--X3 into nibbles in the same register
MOVOA X0, X5
PAND X6, X5
PSRLL $1, X5
MOVOA X1, X4
PAND X7, X4
PADDL X4, X4
PAND X7, X0
PAND X6, X1
POR X4, X0 // X0 = eca86420 (low crumbs)
POR X5, X1 // X1 = fdb97531 (high crumbs)
MOVOA X2, X5
PAND X6, X5
PSRLL $1, X5
MOVOA X3, X4
PAND X7, X4
PADDL X4, X4
PAND X7, X2
PAND X6, X3
POR X4, X2 // X0 = eca86420 (low crumbs)
POR X5, X3 // X1 = fdb97531 (high crumbs)
MOVQ magic<>+16(SB), X7
PSHUFD $0x55, X7, X6 // 0xcccccccc
PSHUFD $0x00, X7, X7 // 0x33333333
MOVOA X0, X5
PAND X6, X5
PSRLL $2, X5
MOVOA X2, X4
PAND X7, X4
PSLLL $2, X4
PAND X7, X0
PAND X6, X2
POR X4, X0 // X0 = c840
POR X5, X2 // X2 = ea62
MOVOA X1, X5
PAND X6, X5
PSRLL $2, X5
MOVOA X3, X4
PAND X7, X4
PSLLL $2, X4
PAND X7, X1
PAND X6, X3
POR X4, X1 // X1 = d951
POR X5, X3 // X3 = fb73
MOVD magic<>+24(SB), X7
PSHUFD $0x00, X7, X7 // 0x0f0f0f0f
// pre-shuffle nibbles
MOVOA X2, X5
PUNPCKLBW X3, X2 // X2 = fbea7362 (3:2:1:0)
PUNPCKHBW X3, X5 // X5 = fbea7362 (7:6:5:4)
MOVOA X0, X3
PUNPCKLBW X1, X0 // X0 = d9c85140 (3:2:1:0)
PUNPCKHBW X1, X3 // X4 = d9c85140 (7:6:5:4)
MOVOA X0, X1
PUNPCKLWL X2, X0 // X0 = fbead9c873625140 (1:0)
PUNPCKHWL X2, X1 // X1 = fbead9c873625140 (3:2)
MOVOA X3, X2
PUNPCKLWL X5, X2 // X2 = fbead9c873625140 (5:4)
PUNPCKHWL X5, X3 // X3 = fbead9c873625140 (7:6)
// pull high and low nibbles and reduce once
MOVOA X0, X4
PSRLL $4, X4
PAND X7, X0 // X0 = ba983210 (1:0)
PAND X7, X4 // X4 = fedc7654 (1:0)
MOVOA X2, X6
PSRLL $4, X2
PAND X7, X6 // X6 = ba983210 (5:4)
PAND X7, X2 // X2 = fedc7654 (5:4)
PADDB X6, X0 // X0 = ba983210 (1:0)
PADDB X4, X2 // X2 = fedc7654 (1:0)
MOVOA X1, X4
PSRLL $4, X4
PAND X7, X1 // X1 = ba983210 (3:2)
PAND X7, X4 // X4 = fedc7654 (3:2)
MOVOA X3, X6
PSRLL $4, X3
PAND X7, X6 // X6 = ba983210 (7:6)
PAND X7, X3 // X3 = fedc7654 (7:6)
PADDB X6, X1 // X1 = ba983210 (3:2)
PADDB X4, X3 // X3 = fedc7654 (3:2)
// unpack one last time
MOVOA X0, X4
PUNPCKLLQ X2, X0 // X0 = fedcba9876543210 (0)
PUNPCKHLQ X2, X4 // X4 = fedcba9876543210 (1)
MOVOA X1, X5
PUNPCKLLQ X3, X1 // X1 = fedcba9876543210 (2)
PUNPCKHLQ X3, X5 // X5 = fedcba9876543210 (3)
// add to counters
PXOR X7, X7 // zero register
ACCUM( X8, X9, X0)
ACCUM(X10, X11, X4)
ACCUM(X12, X13, X1)
ACCUM(X14, X15, X5)
// constants for processing the tail
endvec: MOVQ magic<>+0(SB), X6 // bit position mask
PSHUFD $0x44, X6, X6 // broadcast into both qwords
PXOR X0, X0 // counter registers
PXOR X1, X1
PXOR X2, X2
PXOR X3, X3
// process tail, 4 bytes at a time
SUBL $8-16*16, CX // 8 bytes left to process?
JLT tail1
tail8: MOVL 0(SI), X4
COUNT4(X0, X1)
MOVL 4(SI), X4
COUNT4(X2, X3)
ADDQ $8, SI
SUBL $8, CX
JGE tail8
// process remaining 0--7 byte
tail1: SUBL $-8, CX // anything left to process?
JLE end
MOVQ (SI), X5 // load 8 bytes from buffer. Note that
// buffer is aligned to 8 byte here
MOVQ $window<>+16(SB), AX // load window address
SUBQ CX, AX // adjust mask
MOVQ (AX), X7 // load window mask
PANDN X5, X7 // and mask out the desired bytes
// process rest
MOVOA X7, X4
COUNT4(X0, X1)
PSRLO $4, X7
MOVOA X7, X4
COUNT4(X2, X3)
// add tail to counters
end: PXOR X7, X7 // zero register
MOVOA X0, X4
PUNPCKLBW X7, X0
PUNPCKHBW X7, X4
PADDW X0, X8
PADDW X4, X9
MOVOA X1, X4
PUNPCKLBW X7, X1
PUNPCKHBW X7, X4
PADDW X1, X10
PADDW X4, X11
MOVOA X2, X4
PUNPCKLBW X7, X2
PUNPCKHBW X7, X4
PADDW X2, X12
PADDW X4, X13
MOVOA X3, X4
PUNPCKLBW X7, X3
PUNPCKHBW X7, X4
PADDW X3, X14
PADDW X4, X15
CALL *BX
RET
// buffer is short, do just head/tail processing
runt: SUBL $8, CX // 8 bytes left to process?
JLT runt1
// process runt 8 bytes at a time
runt8: MOVL 0(SI), X4
COUNT4(X8, X10)
MOVL 4(SI), X4
COUNT4(X12, X14)
ADDQ $8, SI
SUBL $8, CX
JGE runt8
// process remaining 0--7 byte
// while making sure we don't get a page fault
runt1: ADDL $8, CX // anything left to process?
JLE runt_accum
MOVL SI, AX
ANDL $7, AX // offset from 8 byte alignment
LEAL (AX)(CX*1), DX // length of buffer plus alignment
SHLL $3, CX // remaining length in bits
XORQ R9, R9
BTSQ CX, R9
DECQ R9 // mask of bits where R8 is in range
CMPL DX, $8 // if this exceeds the alignment boundary
JGT crossrunt1 // we can safely load directly
ANDQ $~7, SI // align buffer to 8 bytes
MOVQ (SI), R8 // and and load 8 bytes from buffer
LEAL (AX*8), CX // offset from 8 byte alignment in bits
SHRQ CX, R8 // buffer starting from the beginning
JMP dorunt1
crossrunt1:
MOVQ (SI), R8 // load 8 bytes from unaligned buffer
dorunt1:
ANDQ R9, R8 // mask out bytes behind the buffer
MOVL R8, X4
SHRQ $32, R8
COUNT4(X8, X10)
MOVL R8, X4
COUNT4(X12, X14)
// move tail to counters and perform final accumulation
runt_accum:
MOVOA X8, X9
PUNPCKLBW X7, X8
PUNPCKHBW X7, X9
MOVOA X10, X11
PUNPCKLBW X7, X10
PUNPCKHBW X7, X11
MOVOA X12, X13
PUNPCKLBW X7, X12
PUNPCKHBW X7, X13
MOVOA X14, X15
PUNPCKLBW X7, X14
PUNPCKHBW X7, X15
CALL *BX
RET
// zero-extend dwords in X trashing X, X4, and X5. Add the low half
// dwords to a*8(DI) and the high half to (a+2)*8(DI).
// Assumes X7 == 0.
#define ACCUMQ(a, X) \
MOVOA X, X4 \
PUNPCKLLQ X7, X \
PUNPCKHLQ X7, X4 \
MOVOU (a)*8(DI), X5 \
PADDQ X, X5 \
MOVOU X5, (a)*8(DI) \
MOVOU (a+2)*8(DI), X5 \
PADDQ X4, X5 \
MOVOU X5, (a+2)*8(DI)
// zero-extend words in X to qwords and add to a*8(DI) to (a+7)*8(DI).
// Trashes X4, X5, and X6. Assumes X7 == 0 an X8 <= X <= X15.
#define ACCUMO(a, X) \
MOVOA X, X6 \
PUNPCKLWL X7, X6 \
PUNPCKHWL X7, X \
ACCUMQ(a, X6) \
ACCUMQ(a+4, X)
// zero-extend words in X and Y to dwords, sum them, and move the
// halves back into X and Y. Assumes X7 == 0. Trashes X4, X5.
#define FOLDW(X, Y) \
MOVOA X, X4 \
PUNPCKLWL X7, X \
PUNPCKHWL X7, X4 \
MOVOA Y, X5 \
PUNPCKLWL X7, X5 \
PUNPCKHWL X7, Y \
PADDL X5, X \
PADDL X4, Y
// Count8 accumulation function. Accumulates words X8--X15 into
// 8 qword counters at (DI). Assumes X7 == 0. Trashes X4--X15.
TEXT accum8<>(SB), NOSPLIT, $0-0
FOLDW(X8, X12)
FOLDW(X9, X13)
FOLDW(X10, X14)
FOLDW(X11, X15)
PADDL X10, X8
PADDL X11, X9
PADDL X14, X12
PADDL X15, X13
PADDL X9, X8
ACCUMQ(0, X8)
PADDL X13, X12
ACCUMQ(4, X12)
RET
// Count16 accumulation function. Accumulates words X8--X15 into
// 16 qword counters at (DI). Assumes X7 == 0. Trashes X4--X15.
TEXT accum16<>(SB), NOSPLIT, $0-0
FOLDW(X8, X12)
FOLDW(X9, X13)
FOLDW(X10, X14)
FOLDW(X11, X15)
PADDL X10, X8
ACCUMQ(0, X8)
PADDL X14, X12
ACCUMQ(4, X12)
PADDL X11, X9
ACCUMQ(8, X9)
PADDL X15, X13
ACCUMQ(12, X13)
RET
// Count32 accumulation function. Accumulates words X8--X15 into
// 32 qword counters at (DI). Assumes X7 == 0. Trashes X4--X15.
TEXT accum32<>(SB), NOSPLIT, $0-0
FOLDW(X8, X12)
ACCUMQ(0, X8)
ACCUMQ(4, X12)
FOLDW(X9, X13)
ACCUMQ(8, X9)
ACCUMQ(12, X13)
FOLDW(X10, X14)
ACCUMQ(16, X10)
ACCUMQ(20, X14)
FOLDW(X11, X15)
ACCUMQ(24, X11)
ACCUMQ(28, X15)
RET
// Count64 accumulation function. Accumulates words X8--X15 into
// 64 qword counters at (DI). Assumes X7 == 0. Trashes X4--X15.
TEXT accum64<>(SB), NOSPLIT, $0-0
ACCUMO(0, X8)
ACCUMO(8, X9)
ACCUMO(16, X10)
ACCUMO(24, X11)
ACCUMO(32, X12)
ACCUMO(40, X13)
ACCUMO(48, X14)
ACCUMO(56, X15)
RET
// func count8sse2(counts *[8]int, buf []uint8)
TEXT ·count8sse2(SB), 0, $0-32
MOVQ counts+0(FP), DI
MOVQ buf_base+8(FP), SI // SI = &buf[0]
MOVQ buf_len+16(FP), CX // CX = len(buf)
MOVQ $accum8<>(SB), BX
CALL countsse2<>(SB)
RET
// func count16sse2(counts *[16]int, buf []uint16)
TEXT ·count16sse2(SB), 0, $0-32
MOVQ counts+0(FP), DI
MOVQ buf_base+8(FP), SI // SI = &buf[0]
MOVQ buf_len+16(FP), CX // CX = len(buf)
MOVQ $accum16<>(SB), BX
SHLQ $1, CX // count in bytes
CALL countsse2<>(SB)
RET
// func count32sse2(counts *[32]int, buf []uint32)
TEXT ·count32sse2(SB), 0, $0-32
MOVQ counts+0(FP), DI
MOVQ buf_base+8(FP), SI // SI = &buf[0]
MOVQ buf_len+16(FP), CX // CX = len(buf)
MOVQ $accum32<>(SB), BX
SHLQ $2, CX // count in bytes
CALL countsse2<>(SB)
RET
// func count64sse2(counts *[64]int, buf []uint64)
TEXT ·count64sse2(SB), 0, $0-32
MOVQ counts+0(FP), DI
MOVQ buf_base+8(FP), SI // SI = &buf[0]
MOVQ buf_len+16(FP), CX // CX = len(buf)
MOVQ $accum64<>(SB), BX
SHLQ $3, CX // count in bytes
CALL countsse2<>(SB)
RET