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
|
/* Instruction cache flushing for hppa */
/*
* Copyright 1995-2017 Bruno Haible <bruno@clisp.org>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#ifdef __hppa64__
/* Tell GCC not to clobber the registers that are call-saved in the HP C
calling convention. */
register long arg0 __asm__("r26");
register long arg1 __asm__("r25");
register long arg2 __asm__("r24");
register long arg3 __asm__("r23");
register long arg4 __asm__("r22");
register long arg5 __asm__("r21");
register long arg6 __asm__("r20");
register long arg7 __asm__("r19");
#endif
/*
* This assumes that the range [first_addr..last_addr] lies in at most two
* cache lines.
*/
void __TR_clear_cache (char* first_addr, char* last_addr)
{
register int tmp1;
register int tmp2;
/* Flush the relevant data cache lines. (Yes, this is needed. I tried it.) */
asm volatile ("fdc 0(0,%0)"
"\n\t" "fdc 0(0,%1)"
"\n\t" "sync"
:
: "r" (first_addr), "r" (last_addr)
);
/* Flush the relevant instruction cache lines. */
asm volatile ("mfsp %%sr0,%1"
"\n\t" "ldsid (0,%4),%0"
"\n\t" "mtsp %0,%%sr0"
"\n\t" "fic 0(%%sr0,%2)"
"\n\t" "fic 0(%%sr0,%3)"
"\n\t" "sync"
"\n\t" "mtsp %1,%%sr0"
"\n\t" "nop"
"\n\t" "nop"
"\n\t" "nop"
"\n\t" "nop"
"\n\t" "nop"
"\n\t" "nop"
: "=r" (tmp1), "=r" (tmp2)
: "r" (first_addr), "r" (last_addr), "r" (first_addr)
);
}
|