1*57a4bec3Smiod /* $OpenBSD: cpustate.h,v 1.11 2015/09/27 18:17:08 miod Exp $ */ 2f58c7388Spefo 3f58c7388Spefo /* 4f58c7388Spefo * Copyright (c) 2002-2003 Opsycon AB (www.opsycon.se / www.opsycon.com) 5f58c7388Spefo * 6f58c7388Spefo * Redistribution and use in source and binary forms, with or without 7f58c7388Spefo * modification, are permitted provided that the following conditions 8f58c7388Spefo * are met: 9f58c7388Spefo * 1. Redistributions of source code must retain the above copyright 10f58c7388Spefo * notice, this list of conditions and the following disclaimer. 11f58c7388Spefo * 2. Redistributions in binary form must reproduce the above copyright 12f58c7388Spefo * notice, this list of conditions and the following disclaimer in the 13f58c7388Spefo * documentation and/or other materials provided with the distribution. 14f58c7388Spefo * 15f58c7388Spefo * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS 16f58c7388Spefo * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 17f58c7388Spefo * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 18f58c7388Spefo * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY 19f58c7388Spefo * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 20f58c7388Spefo * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 21f58c7388Spefo * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 22f58c7388Spefo * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 23f58c7388Spefo * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 24f58c7388Spefo * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 25f58c7388Spefo * SUCH DAMAGE. 26f58c7388Spefo * 27f58c7388Spefo */ 28f58c7388Spefo 29f58c7388Spefo #define KERN_REG_SIZE (NUMSAVEREGS * REGSZ) 30f58c7388Spefo #define KERN_EXC_FRAME_SIZE (CF_SZ + KERN_REG_SIZE + 16) 31f58c7388Spefo 32f58c7388Spefo #define SAVE_REG(reg, offs, base, bo) \ 33f58c7388Spefo REG_S reg, bo + (REGSZ * offs) (base) 34f58c7388Spefo 35f58c7388Spefo #define RESTORE_REG(reg, offs, base, bo) \ 36f58c7388Spefo REG_L reg, bo + (REGSZ * offs) (base) 37f58c7388Spefo 38f58c7388Spefo /* 39f58c7388Spefo * This macro saves the 'scratch' cpu state on stack. 40f58c7388Spefo * Macros are generic so no 'special' instructions! 41f58c7388Spefo * a0 will have a pointer to the 'frame' on return. 42f58c7388Spefo * a1 will have saved STATUS_REG on return. 43f58c7388Spefo * a3 will have the exception pc on 'return'. 44f58c7388Spefo * No traps, no interrupts if frame = k1 or k0! 454a63270fSpefo * Temp regs are saved with their register number so 464a63270fSpefo * branch emulation etc works properly. 47f58c7388Spefo */ 48f58c7388Spefo #define SAVE_CPU(frame, bo) \ 49f58c7388Spefo SAVE_REG(AT, AST, frame, bo) ;\ 50f58c7388Spefo SAVE_REG(v0, V0, frame, bo) ;\ 51f58c7388Spefo SAVE_REG(v1, V1, frame, bo) ;\ 52f58c7388Spefo SAVE_REG(a0, A0, frame, bo) ;\ 53f58c7388Spefo SAVE_REG(a1, A1, frame, bo) ;\ 54f58c7388Spefo SAVE_REG(a2, A2, frame, bo) ;\ 55f58c7388Spefo SAVE_REG(a3, A3, frame, bo) ;\ 563b05e32eSmiod MFC0 a0, COP_0_CAUSE_REG ;\ 57*57a4bec3Smiod MFC0_HAZARD ;\ 58b37e871eSmiod SAVE_REG(a4, A4, frame, bo) ;\ 59b37e871eSmiod SAVE_REG(a5, A5, frame, bo) ;\ 603b05e32eSmiod MFC0 a1, COP_0_STATUS_REG ;\ 61*57a4bec3Smiod MFC0_HAZARD ;\ 62b37e871eSmiod SAVE_REG(a6, A6, frame, bo) ;\ 63b37e871eSmiod SAVE_REG(a7, A7, frame, bo) ;\ 643b05e32eSmiod PRE_MFC0_ADDR_HAZARD ;\ 653b05e32eSmiod DMFC0 a2, COP_0_BAD_VADDR ;\ 66*57a4bec3Smiod MFC0_HAZARD ;\ 67b37e871eSmiod SAVE_REG(t0, T0, frame, bo) ;\ 68b37e871eSmiod SAVE_REG(t1, T1, frame, bo) ;\ 693b05e32eSmiod DMFC0 a3, COP_0_EXC_PC ;\ 70*57a4bec3Smiod MFC0_HAZARD ;\ 71b37e871eSmiod SAVE_REG(t2, T2, frame, bo) ;\ 72b37e871eSmiod SAVE_REG(t3, T3, frame, bo) ;\ 73f58c7388Spefo SAVE_REG(t8, T8, frame, bo) ;\ 74f58c7388Spefo SAVE_REG(t9, T9, frame, bo) ;\ 75f58c7388Spefo SAVE_REG(gp, GP, frame, bo) ;\ 76f58c7388Spefo SAVE_REG(ra, RA, frame, bo) ;\ 77f58c7388Spefo mflo v0 ;\ 78f58c7388Spefo mfhi v1 ;\ 79f58c7388Spefo SAVE_REG(v0, MULLO, frame, bo) ;\ 80f58c7388Spefo SAVE_REG(v1, MULHI, frame, bo) ;\ 81f58c7388Spefo SAVE_REG(a0, CAUSE, frame, bo) ;\ 82f58c7388Spefo SAVE_REG(a1, SR, frame, bo) ;\ 83f58c7388Spefo SAVE_REG(a2, BADVADDR, frame, bo) ;\ 84f58c7388Spefo SAVE_REG(a3, PC, frame, bo) ;\ 85f58c7388Spefo SAVE_REG(sp, SP, frame, bo) ;\ 86ed9f9aaaSpefo PTR_ADDU a0, frame, bo ;\ 8765ca1596Ssyuu GET_CPU_INFO(v0, v1) ;\ 8891b9a95cSmiod lw a2, CI_IPL(v0) ;\ 89f58c7388Spefo SAVE_REG(a2, CPL, frame, bo) 90f58c7388Spefo 91f58c7388Spefo /* 92f58c7388Spefo * Save 'callee save' registers in frame to aid DDB. 93f58c7388Spefo */ 94f58c7388Spefo #define SAVE_CPU_SREG(frame, bo) \ 95f58c7388Spefo SAVE_REG(s0, S0, frame, bo) ;\ 96f58c7388Spefo SAVE_REG(s1, S1, frame, bo) ;\ 97f58c7388Spefo SAVE_REG(s2, S2, frame, bo) ;\ 98f58c7388Spefo SAVE_REG(s3, S3, frame, bo) ;\ 99f58c7388Spefo SAVE_REG(s4, S4, frame, bo) ;\ 100f58c7388Spefo SAVE_REG(s5, S5, frame, bo) ;\ 101f58c7388Spefo SAVE_REG(s6, S6, frame, bo) ;\ 102f58c7388Spefo SAVE_REG(s7, S7, frame, bo) ;\ 103f58c7388Spefo SAVE_REG(s8, S8, frame, bo) 104f58c7388Spefo 105f58c7388Spefo /* 106f58c7388Spefo * Restore cpu state. When called a0 = EXC_PC. 107f58c7388Spefo */ 108f58c7388Spefo #define RESTORE_CPU(frame, bo) \ 109f58c7388Spefo RESTORE_REG(t1, SR, frame, bo) ;\ 110f58c7388Spefo RESTORE_REG(t2, MULLO, frame, bo) ;\ 111f58c7388Spefo RESTORE_REG(t3, MULHI, frame, bo) ;\ 1123b05e32eSmiod MTC0 t1, COP_0_STATUS_REG ;\ 1133b05e32eSmiod MTC0_SR_IE_HAZARD ;\ 114f58c7388Spefo mtlo t2 ;\ 115f58c7388Spefo mthi t3 ;\ 1163b05e32eSmiod DMTC0 a0, COP_0_EXC_PC ;\ 1173b05e32eSmiod MTC0_HAZARD ;\ 118f58c7388Spefo RESTORE_REG(AT, AST, frame, bo) ;\ 119f58c7388Spefo RESTORE_REG(v0, V0, frame, bo) ;\ 120f58c7388Spefo RESTORE_REG(v1, V1, frame, bo) ;\ 121f58c7388Spefo RESTORE_REG(a0, A0, frame, bo) ;\ 122f58c7388Spefo RESTORE_REG(a1, A1, frame, bo) ;\ 123f58c7388Spefo RESTORE_REG(a2, A2, frame, bo) ;\ 124f58c7388Spefo RESTORE_REG(a3, A3, frame, bo) ;\ 125b37e871eSmiod RESTORE_REG(a4, A4, frame, bo) ;\ 126b37e871eSmiod RESTORE_REG(a5, A5, frame, bo) ;\ 127b37e871eSmiod RESTORE_REG(a6, A6, frame, bo) ;\ 128b37e871eSmiod RESTORE_REG(a7, A7, frame, bo) ;\ 129b37e871eSmiod RESTORE_REG(t0, T0, frame, bo) ;\ 130b37e871eSmiod RESTORE_REG(t1, T1, frame, bo) ;\ 131b37e871eSmiod RESTORE_REG(t2, T2, frame, bo) ;\ 132b37e871eSmiod RESTORE_REG(t3, T3, frame, bo) ;\ 133f58c7388Spefo RESTORE_REG(t8, T8, frame, bo) ;\ 134f58c7388Spefo RESTORE_REG(t9, T9, frame, bo) ;\ 135f58c7388Spefo RESTORE_REG(gp, GP, frame, bo) ;\ 136f58c7388Spefo RESTORE_REG(ra, RA, frame, bo) 137f58c7388Spefo 138f58c7388Spefo /* 139f58c7388Spefo * Restore 'callee save' registers 140f58c7388Spefo */ 141f58c7388Spefo #define RESTORE_CPU_SREG(frame, bo) \ 142f58c7388Spefo RESTORE_REG(s0, S0, frame, bo) ;\ 143f58c7388Spefo RESTORE_REG(s1, S1, frame, bo) ;\ 144f58c7388Spefo RESTORE_REG(s2, S2, frame, bo) ;\ 145f58c7388Spefo RESTORE_REG(s3, S3, frame, bo) ;\ 146f58c7388Spefo RESTORE_REG(s4, S4, frame, bo) ;\ 147f58c7388Spefo RESTORE_REG(s5, S5, frame, bo) ;\ 148f58c7388Spefo RESTORE_REG(s6, S6, frame, bo) ;\ 149f58c7388Spefo RESTORE_REG(s7, S7, frame, bo) ;\ 150f58c7388Spefo RESTORE_REG(s8, S8, frame, bo) 151