1/* 2 * "memcpy" implementation of SuperH 3 * 4 * Copyright (C) 1999 Niibe Yutaka 5 * Copyright (c) 2002 STMicroelectronics Ltd 6 * Modified from memcpy.S and micro-optimised for SH4 7 * Stuart Menefy (stuart.menefy@st.com) 8 * 9 */ 10#include <linux/linkage.h> 11 12/* 13 * void *memcpy(void *dst, const void *src, size_t n); 14 * 15 * It is assumed that there is no overlap between src and dst. 16 * If there is an overlap, then the results are undefined. 17 */ 18 19 ! 20 ! GHIJ KLMN OPQR --> ...G HIJK LMNO PQR. 21 ! 22 23 ! Size is 16 or greater, and may have trailing bytes 24 25 .balign 32 26.Lcase1: 27 ! Read a long word and write a long word at once 28 ! At the start of each iteration, r7 contains last long load 29 add #-1,r5 ! 79 EX 30 mov r4,r2 ! 5 MT (0 cycles latency) 31 32 mov.l @(r0,r5),r7 ! 21 LS (2 cycles latency) 33 add #-4,r5 ! 50 EX 34 35 add #7,r2 ! 79 EX 36 ! 37#ifdef CONFIG_CPU_LITTLE_ENDIAN 38 ! 6 cycles, 4 bytes per iteration 393: mov.l @(r0,r5),r1 ! 21 LS (latency=2) ! NMLK 40 mov r7, r3 ! 5 MT (latency=0) ! RQPO 41 42 cmp/hi r2,r0 ! 57 MT 43 shll16 r3 ! 103 EX 44 45 mov r1,r6 ! 5 MT (latency=0) 46 shll8 r3 ! 102 EX ! Oxxx 47 48 shlr8 r6 ! 106 EX ! xNML 49 mov r1, r7 ! 5 MT (latency=0) 50 51 or r6,r3 ! 82 EX ! ONML 52 bt/s 3b ! 109 BR 53 54 mov.l r3,@-r0 ! 30 LS 55#else 563: mov.l @(r0,r5),r1 ! 21 LS (latency=2) ! KLMN 57 mov r7,r3 ! 5 MT (latency=0) ! OPQR 58 59 cmp/hi r2,r0 ! 57 MT 60 shlr16 r3 ! 107 EX 61 62 shlr8 r3 ! 106 EX ! xxxO 63 mov r1,r6 ! 5 MT (latency=0) 64 65 shll8 r6 ! 102 EX ! LMNx 66 mov r1,r7 ! 5 MT (latency=0) 67 68 or r6,r3 ! 82 EX ! LMNO 69 bt/s 3b ! 109 BR 70 71 mov.l r3,@-r0 ! 30 LS 72#endif 73 ! Finally, copy a byte at once, if necessary 74 75 add #4,r5 ! 50 EX 76 cmp/eq r4,r0 ! 54 MT 77 78 add #-6,r2 ! 50 EX 79 bt 9f ! 109 BR 80 818: cmp/hi r2,r0 ! 57 MT 82 mov.b @(r0,r5),r1 ! 20 LS (latency=2) 83 84 bt/s 8b ! 109 BR 85 86 mov.b r1,@-r0 ! 29 LS 87 889: rts 89 nop 90 91 92 ! 93 ! GHIJ KLMN OPQR --> .GHI JKLM NOPQ R... 94 ! 95 96 ! Size is 16 or greater, and may have trailing bytes 97 98 .balign 32 99.Lcase3: 100 ! Read a long word and write a long word at once 101 ! At the start of each iteration, r7 contains last long load 102 add #-3,r5 ! 79 EX 103 mov r4,r2 ! 5 MT (0 cycles latency) 104 105 mov.l @(r0,r5),r7 ! 21 LS (2 cycles latency) 106 add #-4,r5 ! 50 EX 107 108 add #7,r2 ! 79 EX 109 ! 110#ifdef CONFIG_CPU_LITTLE_ENDIAN 111 ! 6 cycles, 4 bytes per iteration 1123: mov.l @(r0,r5),r1 ! 21 LS (latency=2) ! NMLK 113 mov r7, r3 ! 5 MT (latency=0) ! RQPO 114 115 cmp/hi r2,r0 ! 57 MT 116 shll8 r3 ! 102 EX ! QPOx 117 118 mov r1,r6 ! 5 MT (latency=0) 119 shlr16 r6 ! 107 EX 120 121 shlr8 r6 ! 106 EX ! xxxN 122 mov r1, r7 ! 5 MT (latency=0) 123 124 or r6,r3 ! 82 EX ! QPON 125 bt/s 3b ! 109 BR 126 127 mov.l r3,@-r0 ! 30 LS 128#else 1293: mov r1,r3 ! OPQR 130 shlr8 r3 ! xOPQ 131 mov.l @(r0,r5),r1 ! KLMN 132 mov r1,r6 133 shll16 r6 134 shll8 r6 ! Nxxx 135 or r6,r3 ! NOPQ 136 cmp/hi r2,r0 137 bt/s 3b 138 mov.l r3,@-r0 139#endif 140 141 ! Finally, copy a byte at once, if necessary 142 143 add #6,r5 ! 50 EX 144 cmp/eq r4,r0 ! 54 MT 145 146 add #-6,r2 ! 50 EX 147 bt 9f ! 109 BR 148 1498: cmp/hi r2,r0 ! 57 MT 150 mov.b @(r0,r5),r1 ! 20 LS (latency=2) 151 152 bt/s 8b ! 109 BR 153 154 mov.b r1,@-r0 ! 29 LS 155 1569: rts 157 nop 158 159ENTRY(memcpy) 160 161 ! Calculate the invariants which will be used in the remainder 162 ! of the code: 163 ! 164 ! r4 --> [ ... ] DST [ ... ] SRC 165 ! [ ... ] [ ... ] 166 ! : : 167 ! r0 --> [ ... ] r0+r5 --> [ ... ] 168 ! 169 ! 170 171 ! Short circuit the common case of src, dst and len being 32 bit aligned 172 ! and test for zero length move 173 174 mov r6, r0 ! 5 MT (0 cycle latency) 175 or r4, r0 ! 82 EX 176 177 or r5, r0 ! 82 EX 178 tst r6, r6 ! 86 MT 179 180 bt/s 99f ! 111 BR (zero len) 181 tst #3, r0 ! 87 MT 182 183 mov r4, r0 ! 5 MT (0 cycle latency) 184 add r6, r0 ! 49 EX 185 186 mov #16, r1 ! 6 EX 187 bt/s .Lcase00 ! 111 BR (aligned) 188 189 sub r4, r5 ! 75 EX 190 191 ! Arguments are not nicely long word aligned or zero len. 192 ! Check for small copies, and if so do a simple byte at a time copy. 193 ! 194 ! Deciding on an exact value of 'small' is not easy, as the point at which 195 ! using the optimised routines become worthwhile varies (these are the 196 ! cycle counts for differnet sizes using byte-at-a-time vs. optimised): 197 ! size byte-at-time long word byte 198 ! 16 42 39-40 46-50 50-55 199 ! 24 58 43-44 54-58 62-67 200 ! 36 82 49-50 66-70 80-85 201 ! However the penalty for getting it 'wrong' is much higher for long word 202 ! aligned data (and this is more common), so use a value of 16. 203 204 cmp/gt r6,r1 ! 56 MT 205 206 add #-1,r5 ! 50 EX 207 bf/s 6f ! 108 BR (not small) 208 209 mov r5, r3 ! 5 MT (latency=0) 210 shlr r6 ! 104 EX 211 212 mov.b @(r0,r5),r1 ! 20 LS (latency=2) 213 bf/s 4f ! 111 BR 214 215 add #-1,r3 ! 50 EX 216 tst r6, r6 ! 86 MT 217 218 bt/s 98f ! 110 BR 219 mov.b r1,@-r0 ! 29 LS 220 221 ! 4 cycles, 2 bytes per iteration 2223: mov.b @(r0,r5),r1 ! 20 LS (latency=2) 223 2244: mov.b @(r0,r3),r2 ! 20 LS (latency=2) 225 dt r6 ! 67 EX 226 227 mov.b r1,@-r0 ! 29 LS 228 bf/s 3b ! 111 BR 229 230 mov.b r2,@-r0 ! 29 LS 23198: 232 rts 233 nop 234 23599: rts 236 mov r4, r0 237 238 ! Size is not small, so its worthwhile looking for optimisations. 239 ! First align destination to a long word boundary. 240 ! 241 ! r5 = normal value -1 242 2436: tst #3, r0 ! 87 MT 244 mov #3, r3 ! 6 EX 245 246 bt/s 2f ! 111 BR 247 and r0,r3 ! 78 EX 248 249 ! 3 cycles, 1 byte per iteration 2501: dt r3 ! 67 EX 251 mov.b @(r0,r5),r1 ! 19 LS (latency=2) 252 253 add #-1, r6 ! 79 EX 254 bf/s 1b ! 109 BR 255 256 mov.b r1,@-r0 ! 28 LS 257 2582: add #1, r5 ! 79 EX 259 260 ! Now select the appropriate bulk transfer code based on relative 261 ! alignment of src and dst. 262 263 mov r0, r3 ! 5 MT (latency=0) 264 265 mov r5, r0 ! 5 MT (latency=0) 266 tst #1, r0 ! 87 MT 267 268 bf/s 1f ! 111 BR 269 mov #64, r7 ! 6 EX 270 271 ! bit 0 clear 272 273 cmp/ge r7, r6 ! 55 MT 274 275 bt/s 2f ! 111 BR 276 tst #2, r0 ! 87 MT 277 278 ! small 279 bt/s .Lcase0 280 mov r3, r0 281 282 bra .Lcase2 283 nop 284 285 ! big 2862: bt/s .Lcase0b 287 mov r3, r0 288 289 bra .Lcase2b 290 nop 291 292 ! bit 0 set 2931: tst #2, r0 ! 87 MT 294 295 bt/s .Lcase1 296 mov r3, r0 297 298 bra .Lcase3 299 nop 300 301 302 ! 303 ! GHIJ KLMN OPQR --> GHIJ KLMN OPQR 304 ! 305 306 ! src, dst and size are all long word aligned 307 ! size is non-zero 308 309 .balign 32 310.Lcase00: 311 mov #64, r1 ! 6 EX 312 mov r5, r3 ! 5 MT (latency=0) 313 314 cmp/gt r6, r1 ! 56 MT 315 add #-4, r5 ! 50 EX 316 317 bf .Lcase00b ! 108 BR (big loop) 318 shlr2 r6 ! 105 EX 319 320 shlr r6 ! 104 EX 321 mov.l @(r0, r5), r1 ! 21 LS (latency=2) 322 323 bf/s 4f ! 111 BR 324 add #-8, r3 ! 50 EX 325 326 tst r6, r6 ! 86 MT 327 bt/s 5f ! 110 BR 328 329 mov.l r1,@-r0 ! 30 LS 330 331 ! 4 cycles, 2 long words per iteration 3323: mov.l @(r0, r5), r1 ! 21 LS (latency=2) 333 3344: mov.l @(r0, r3), r2 ! 21 LS (latency=2) 335 dt r6 ! 67 EX 336 337 mov.l r1, @-r0 ! 30 LS 338 bf/s 3b ! 109 BR 339 340 mov.l r2, @-r0 ! 30 LS 341 3425: rts 343 nop 344 345 346 ! Size is 16 or greater and less than 64, but may have trailing bytes 347 348 .balign 32 349.Lcase0: 350 add #-4, r5 ! 50 EX 351 mov r4, r7 ! 5 MT (latency=0) 352 353 mov.l @(r0, r5), r1 ! 21 LS (latency=2) 354 mov #4, r2 ! 6 EX 355 356 add #11, r7 ! 50 EX 357 tst r2, r6 ! 86 MT 358 359 mov r5, r3 ! 5 MT (latency=0) 360 bt/s 4f ! 111 BR 361 362 add #-4, r3 ! 50 EX 363 mov.l r1,@-r0 ! 30 LS 364 365 ! 4 cycles, 2 long words per iteration 3663: mov.l @(r0, r5), r1 ! 21 LS (latency=2) 367 3684: mov.l @(r0, r3), r2 ! 21 LS (latency=2) 369 cmp/hi r7, r0 370 371 mov.l r1, @-r0 ! 30 LS 372 bt/s 3b ! 109 BR 373 374 mov.l r2, @-r0 ! 30 LS 375 376 ! Copy the final 0-3 bytes 377 378 add #3,r5 ! 50 EX 379 380 cmp/eq r0, r4 ! 54 MT 381 add #-10, r7 ! 50 EX 382 383 bt 9f ! 110 BR 384 385 ! 3 cycles, 1 byte per iteration 3861: mov.b @(r0,r5),r1 ! 19 LS 387 cmp/hi r7,r0 ! 57 MT 388 389 bt/s 1b ! 111 BR 390 mov.b r1,@-r0 ! 28 LS 391 3929: rts 393 nop 394 395 ! Size is at least 64 bytes, so will be going round the big loop at least once. 396 ! 397 ! r2 = rounded up r4 398 ! r3 = rounded down r0 399 400 .balign 32 401.Lcase0b: 402 add #-4, r5 ! 50 EX 403 404.Lcase00b: 405 mov r0, r3 ! 5 MT (latency=0) 406 mov #(~0x1f), r1 ! 6 EX 407 408 and r1, r3 ! 78 EX 409 mov r4, r2 ! 5 MT (latency=0) 410 411 cmp/eq r3, r0 ! 54 MT 412 add #0x1f, r2 ! 50 EX 413 414 bt/s 1f ! 110 BR 415 and r1, r2 ! 78 EX 416 417 ! copy initial words until cache line aligned 418 419 mov.l @(r0, r5), r1 ! 21 LS (latency=2) 420 tst #4, r0 ! 87 MT 421 422 mov r5, r6 ! 5 MT (latency=0) 423 add #-4, r6 ! 50 EX 424 425 bt/s 4f ! 111 BR 426 add #8, r3 ! 50 EX 427 428 tst #0x18, r0 ! 87 MT 429 430 bt/s 1f ! 109 BR 431 mov.l r1,@-r0 ! 30 LS 432 433 ! 4 cycles, 2 long words per iteration 4343: mov.l @(r0, r5), r1 ! 21 LS (latency=2) 435 4364: mov.l @(r0, r6), r7 ! 21 LS (latency=2) 437 cmp/eq r3, r0 ! 54 MT 438 439 mov.l r1, @-r0 ! 30 LS 440 bf/s 3b ! 109 BR 441 442 mov.l r7, @-r0 ! 30 LS 443 444 ! Copy the cache line aligned blocks 445 ! 446 ! In use: r0, r2, r4, r5 447 ! Scratch: r1, r3, r6, r7 448 ! 449 ! We could do this with the four scratch registers, but if src 450 ! and dest hit the same cache line, this will thrash, so make 451 ! use of additional registers. 452 ! 453 ! We also need r0 as a temporary (for movca), so 'undo' the invariant: 454 ! r5: src (was r0+r5) 455 ! r1: dest (was r0) 456 ! this can be reversed at the end, so we don't need to save any extra 457 ! state. 458 ! 4591: mov.l r8, @-r15 ! 30 LS 460 add r0, r5 ! 49 EX 461 462 mov.l r9, @-r15 ! 30 LS 463 mov r0, r1 ! 5 MT (latency=0) 464 465 mov.l r10, @-r15 ! 30 LS 466 add #-0x1c, r5 ! 50 EX 467 468 mov.l r11, @-r15 ! 30 LS 469 470 ! 16 cycles, 32 bytes per iteration 4712: mov.l @(0x00,r5),r0 ! 18 LS (latency=2) 472 add #-0x20, r1 ! 50 EX 473 mov.l @(0x04,r5),r3 ! 18 LS (latency=2) 474 mov.l @(0x08,r5),r6 ! 18 LS (latency=2) 475 mov.l @(0x0c,r5),r7 ! 18 LS (latency=2) 476 mov.l @(0x10,r5),r8 ! 18 LS (latency=2) 477 mov.l @(0x14,r5),r9 ! 18 LS (latency=2) 478 mov.l @(0x18,r5),r10 ! 18 LS (latency=2) 479 mov.l @(0x1c,r5),r11 ! 18 LS (latency=2) 480 movca.l r0,@r1 ! 40 LS (latency=3-7) 481 mov.l r3,@(0x04,r1) ! 33 LS 482 mov.l r6,@(0x08,r1) ! 33 LS 483 mov.l r7,@(0x0c,r1) ! 33 LS 484 485 mov.l r8,@(0x10,r1) ! 33 LS 486 add #-0x20, r5 ! 50 EX 487 488 mov.l r9,@(0x14,r1) ! 33 LS 489 cmp/eq r2,r1 ! 54 MT 490 491 mov.l r10,@(0x18,r1) ! 33 LS 492 bf/s 2b ! 109 BR 493 494 mov.l r11,@(0x1c,r1) ! 33 LS 495 496 mov r1, r0 ! 5 MT (latency=0) 497 498 mov.l @r15+, r11 ! 15 LS 499 sub r1, r5 ! 75 EX 500 501 mov.l @r15+, r10 ! 15 LS 502 cmp/eq r4, r0 ! 54 MT 503 504 bf/s 1f ! 109 BR 505 mov.l @r15+, r9 ! 15 LS 506 507 rts 5081: mov.l @r15+, r8 ! 15 LS 509 sub r4, r1 ! 75 EX (len remaining) 510 511 ! number of trailing bytes is non-zero 512 ! 513 ! invariants restored (r5 already decremented by 4) 514 ! also r1=num bytes remaining 515 516 mov #4, r2 ! 6 EX 517 mov r4, r7 ! 5 MT (latency=0) 518 519 add #0x1c, r5 ! 50 EX (back to -4) 520 cmp/hs r2, r1 ! 58 MT 521 522 bf/s 5f ! 108 BR 523 add #11, r7 ! 50 EX 524 525 mov.l @(r0, r5), r6 ! 21 LS (latency=2) 526 tst r2, r1 ! 86 MT 527 528 mov r5, r3 ! 5 MT (latency=0) 529 bt/s 4f ! 111 BR 530 531 add #-4, r3 ! 50 EX 532 cmp/hs r2, r1 ! 58 MT 533 534 bt/s 5f ! 111 BR 535 mov.l r6,@-r0 ! 30 LS 536 537 ! 4 cycles, 2 long words per iteration 5383: mov.l @(r0, r5), r6 ! 21 LS (latency=2) 539 5404: mov.l @(r0, r3), r2 ! 21 LS (latency=2) 541 cmp/hi r7, r0 542 543 mov.l r6, @-r0 ! 30 LS 544 bt/s 3b ! 109 BR 545 546 mov.l r2, @-r0 ! 30 LS 547 548 ! Copy the final 0-3 bytes 549 5505: cmp/eq r0, r4 ! 54 MT 551 add #-10, r7 ! 50 EX 552 553 bt 9f ! 110 BR 554 add #3,r5 ! 50 EX 555 556 ! 3 cycles, 1 byte per iteration 5571: mov.b @(r0,r5),r1 ! 19 LS 558 cmp/hi r7,r0 ! 57 MT 559 560 bt/s 1b ! 111 BR 561 mov.b r1,@-r0 ! 28 LS 562 5639: rts 564 nop 565 566 ! 567 ! GHIJ KLMN OPQR --> ..GH IJKL MNOP QR.. 568 ! 569 570 .balign 32 571.Lcase2: 572 ! Size is 16 or greater and less then 64, but may have trailing bytes 573 5742: mov r5, r6 ! 5 MT (latency=0) 575 add #-2,r5 ! 50 EX 576 577 mov r4,r2 ! 5 MT (latency=0) 578 add #-4,r6 ! 50 EX 579 580 add #7,r2 ! 50 EX 5813: mov.w @(r0,r5),r1 ! 20 LS (latency=2) 582 583 mov.w @(r0,r6),r3 ! 20 LS (latency=2) 584 cmp/hi r2,r0 ! 57 MT 585 586 mov.w r1,@-r0 ! 29 LS 587 bt/s 3b ! 111 BR 588 589 mov.w r3,@-r0 ! 29 LS 590 591 bra 10f 592 nop 593 594 595 .balign 32 596.Lcase2b: 597 ! Size is at least 64 bytes, so will be going round the big loop at least once. 598 ! 599 ! r2 = rounded up r4 600 ! r3 = rounded down r0 601 602 mov r0, r3 ! 5 MT (latency=0) 603 mov #(~0x1f), r1 ! 6 EX 604 605 and r1, r3 ! 78 EX 606 mov r4, r2 ! 5 MT (latency=0) 607 608 cmp/eq r3, r0 ! 54 MT 609 add #0x1f, r2 ! 50 EX 610 611 add #-2, r5 ! 50 EX 612 bt/s 1f ! 110 BR 613 and r1, r2 ! 78 EX 614 615 ! Copy a short word one at a time until we are cache line aligned 616 ! Normal values: r0, r2, r3, r4 617 ! Unused: r1, r6, r7 618 ! Mod: r5 (=r5-2) 619 ! 620 add #2, r3 ! 50 EX 621 6222: mov.w @(r0,r5),r1 ! 20 LS (latency=2) 623 cmp/eq r3,r0 ! 54 MT 624 625 bf/s 2b ! 111 BR 626 627 mov.w r1,@-r0 ! 29 LS 628 629 ! Copy the cache line aligned blocks 630 ! 631 ! In use: r0, r2, r4, r5 (=r5-2) 632 ! Scratch: r1, r3, r6, r7 633 ! 634 ! We could do this with the four scratch registers, but if src 635 ! and dest hit the same cache line, this will thrash, so make 636 ! use of additional registers. 637 ! 638 ! We also need r0 as a temporary (for movca), so 'undo' the invariant: 639 ! r5: src (was r0+r5) 640 ! r1: dest (was r0) 641 ! this can be reversed at the end, so we don't need to save any extra 642 ! state. 643 ! 6441: mov.l r8, @-r15 ! 30 LS 645 add r0, r5 ! 49 EX 646 647 mov.l r9, @-r15 ! 30 LS 648 mov r0, r1 ! 5 MT (latency=0) 649 650 mov.l r10, @-r15 ! 30 LS 651 add #-0x1e, r5 ! 50 EX 652 653 mov.l r11, @-r15 ! 30 LS 654 655 mov.l r12, @-r15 ! 30 LS 656 657 ! 17 cycles, 32 bytes per iteration 658#ifdef CONFIG_CPU_LITTLE_ENDIAN 6592: mov.w @r5+, r0 ! 14 LS (latency=2) ..JI 660 add #-0x20, r1 ! 50 EX 661 662 mov.l @r5+, r3 ! 15 LS (latency=2) NMLK 663 664 mov.l @r5+, r6 ! 15 LS (latency=2) RQPO 665 shll16 r0 ! 103 EX JI.. 666 667 mov.l @r5+, r7 ! 15 LS (latency=2) 668 xtrct r3, r0 ! 48 EX LKJI 669 670 mov.l @r5+, r8 ! 15 LS (latency=2) 671 xtrct r6, r3 ! 48 EX PONM 672 673 mov.l @r5+, r9 ! 15 LS (latency=2) 674 xtrct r7, r6 ! 48 EX 675 676 mov.l @r5+, r10 ! 15 LS (latency=2) 677 xtrct r8, r7 ! 48 EX 678 679 mov.l @r5+, r11 ! 15 LS (latency=2) 680 xtrct r9, r8 ! 48 EX 681 682 mov.w @r5+, r12 ! 15 LS (latency=2) 683 xtrct r10, r9 ! 48 EX 684 685 movca.l r0,@r1 ! 40 LS (latency=3-7) 686 xtrct r11, r10 ! 48 EX 687 688 mov.l r3, @(0x04,r1) ! 33 LS 689 xtrct r12, r11 ! 48 EX 690 691 mov.l r6, @(0x08,r1) ! 33 LS 692 693 mov.l r7, @(0x0c,r1) ! 33 LS 694 695 mov.l r8, @(0x10,r1) ! 33 LS 696 add #-0x40, r5 ! 50 EX 697 698 mov.l r9, @(0x14,r1) ! 33 LS 699 cmp/eq r2,r1 ! 54 MT 700 701 mov.l r10, @(0x18,r1) ! 33 LS 702 bf/s 2b ! 109 BR 703 704 mov.l r11, @(0x1c,r1) ! 33 LS 705#else 7062: mov.w @(0x1e,r5), r0 ! 17 LS (latency=2) 707 add #-2, r5 ! 50 EX 708 709 mov.l @(0x1c,r5), r3 ! 18 LS (latency=2) 710 add #-4, r1 ! 50 EX 711 712 mov.l @(0x18,r5), r6 ! 18 LS (latency=2) 713 shll16 r0 ! 103 EX 714 715 mov.l @(0x14,r5), r7 ! 18 LS (latency=2) 716 xtrct r3, r0 ! 48 EX 717 718 mov.l @(0x10,r5), r8 ! 18 LS (latency=2) 719 xtrct r6, r3 ! 48 EX 720 721 mov.l @(0x0c,r5), r9 ! 18 LS (latency=2) 722 xtrct r7, r6 ! 48 EX 723 724 mov.l @(0x08,r5), r10 ! 18 LS (latency=2) 725 xtrct r8, r7 ! 48 EX 726 727 mov.l @(0x04,r5), r11 ! 18 LS (latency=2) 728 xtrct r9, r8 ! 48 EX 729 730 mov.w @(0x02,r5), r12 ! 18 LS (latency=2) 731 xtrct r10, r9 ! 48 EX 732 733 movca.l r0,@r1 ! 40 LS (latency=3-7) 734 add #-0x1c, r1 ! 50 EX 735 736 mov.l r3, @(0x1c,r1) ! 33 LS 737 xtrct r11, r10 ! 48 EX 738 739 mov.l r6, @(0x18,r1) ! 33 LS 740 xtrct r12, r11 ! 48 EX 741 742 mov.l r7, @(0x14,r1) ! 33 LS 743 744 mov.l r8, @(0x10,r1) ! 33 LS 745 add #-0x3e, r5 ! 50 EX 746 747 mov.l r9, @(0x0c,r1) ! 33 LS 748 cmp/eq r2,r1 ! 54 MT 749 750 mov.l r10, @(0x08,r1) ! 33 LS 751 bf/s 2b ! 109 BR 752 753 mov.l r11, @(0x04,r1) ! 33 LS 754#endif 755 756 mov.l @r15+, r12 757 mov r1, r0 ! 5 MT (latency=0) 758 759 mov.l @r15+, r11 ! 15 LS 760 sub r1, r5 ! 75 EX 761 762 mov.l @r15+, r10 ! 15 LS 763 cmp/eq r4, r0 ! 54 MT 764 765 bf/s 1f ! 109 BR 766 mov.l @r15+, r9 ! 15 LS 767 768 rts 7691: mov.l @r15+, r8 ! 15 LS 770 771 add #0x1e, r5 ! 50 EX 772 773 ! Finish off a short word at a time 774 ! r5 must be invariant - 2 77510: mov r4,r2 ! 5 MT (latency=0) 776 add #1,r2 ! 50 EX 777 778 cmp/hi r2, r0 ! 57 MT 779 bf/s 1f ! 109 BR 780 781 add #2, r2 ! 50 EX 782 7833: mov.w @(r0,r5),r1 ! 20 LS 784 cmp/hi r2,r0 ! 57 MT 785 786 bt/s 3b ! 109 BR 787 788 mov.w r1,@-r0 ! 29 LS 7891: 790 791 ! 792 ! Finally, copy the last byte if necessary 793 cmp/eq r4,r0 ! 54 MT 794 bt/s 9b 795 add #1,r5 796 mov.b @(r0,r5),r1 797 rts 798 mov.b r1,@-r0 799 800