|  | /* libs/pixelflinger/codeflinger/ARMAssemblerInterface.h | 
|  | ** | 
|  | ** Copyright 2006, The Android Open Source Project | 
|  | ** | 
|  | ** Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | ** you may not use this file except in compliance with the License. | 
|  | ** You may obtain a copy of the License at | 
|  | ** | 
|  | **     http://www.apache.org/licenses/LICENSE-2.0 | 
|  | ** | 
|  | ** Unless required by applicable law or agreed to in writing, software | 
|  | ** distributed under the License is distributed on an "AS IS" BASIS, | 
|  | ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | ** See the License for the specific language governing permissions and | 
|  | ** limitations under the License. | 
|  | */ | 
|  |  | 
|  |  | 
|  | #ifndef ANDROID_ARMASSEMBLER_INTERFACE_H | 
|  | #define ANDROID_ARMASSEMBLER_INTERFACE_H | 
|  |  | 
|  | #include <stdint.h> | 
|  | #include <sys/types.h> | 
|  |  | 
|  | namespace android { | 
|  |  | 
|  | // ---------------------------------------------------------------------------- | 
|  |  | 
|  | class ARMAssemblerInterface | 
|  | { | 
|  | public: | 
|  | virtual ~ARMAssemblerInterface(); | 
|  |  | 
|  | enum { | 
|  | EQ, NE, CS, CC, MI, PL, VS, VC, HI, LS, GE, LT, GT, LE, AL, NV, | 
|  | HS = CS, | 
|  | LO = CC | 
|  | }; | 
|  | enum { | 
|  | S = 1 | 
|  | }; | 
|  | enum { | 
|  | LSL, LSR, ASR, ROR | 
|  | }; | 
|  | enum { | 
|  | ED, FD, EA, FA, | 
|  | IB, IA, DB, DA | 
|  | }; | 
|  | enum { | 
|  | R0, R1, R2, R3, R4, R5, R6, R7, R8, R9, R10, R11, R12, R13, R14, R15, | 
|  | SP = R13, | 
|  | LR = R14, | 
|  | PC = R15 | 
|  | }; | 
|  | enum { | 
|  | #define LIST(rr) L##rr=1<<rr | 
|  | LIST(R0), LIST(R1), LIST(R2), LIST(R3), LIST(R4), LIST(R5), LIST(R6), | 
|  | LIST(R7), LIST(R8), LIST(R9), LIST(R10), LIST(R11), LIST(R12), | 
|  | LIST(R13), LIST(R14), LIST(R15), | 
|  | LIST(SP), LIST(LR), LIST(PC), | 
|  | #undef LIST | 
|  | LSAVED = LR4|LR5|LR6|LR7|LR8|LR9|LR10|LR11 | LLR | 
|  | }; | 
|  |  | 
|  | enum { | 
|  | CODEGEN_ARCH_ARM = 1, CODEGEN_ARCH_MIPS, CODEGEN_ARCH_ARM64, CODEGEN_ARCH_MIPS64 | 
|  | }; | 
|  |  | 
|  | // ----------------------------------------------------------------------- | 
|  | // shifters and addressing modes | 
|  | // ----------------------------------------------------------------------- | 
|  |  | 
|  | // these static versions are used for initializers on LDxx/STxx below | 
|  | static uint32_t    __immed12_pre(int32_t immed12, int W=0); | 
|  | static uint32_t    __immed8_pre(int32_t immed12, int W=0); | 
|  |  | 
|  | virtual bool        isValidImmediate(uint32_t immed) = 0; | 
|  | virtual int         buildImmediate(uint32_t i, uint32_t& rot, uint32_t& imm) = 0; | 
|  |  | 
|  | virtual uint32_t    imm(uint32_t immediate) = 0; | 
|  | virtual uint32_t    reg_imm(int Rm, int type, uint32_t shift) = 0; | 
|  | virtual uint32_t    reg_rrx(int Rm) = 0; | 
|  | virtual uint32_t    reg_reg(int Rm, int type, int Rs) = 0; | 
|  |  | 
|  | // addressing modes... | 
|  | // LDR(B)/STR(B)/PLD | 
|  | // (immediate and Rm can be negative, which indicates U=0) | 
|  | virtual uint32_t    immed12_pre(int32_t immed12, int W=0) = 0; | 
|  | virtual uint32_t    immed12_post(int32_t immed12) = 0; | 
|  | virtual uint32_t    reg_scale_pre(int Rm, int type=0, uint32_t shift=0, int W=0) = 0; | 
|  | virtual uint32_t    reg_scale_post(int Rm, int type=0, uint32_t shift=0) = 0; | 
|  |  | 
|  | // LDRH/LDRSB/LDRSH/STRH | 
|  | // (immediate and Rm can be negative, which indicates U=0) | 
|  | virtual uint32_t    immed8_pre(int32_t immed8, int W=0) = 0; | 
|  | virtual uint32_t    immed8_post(int32_t immed8) = 0; | 
|  | virtual uint32_t    reg_pre(int Rm, int W=0) = 0; | 
|  | virtual uint32_t    reg_post(int Rm) = 0; | 
|  |  | 
|  | // ----------------------------------------------------------------------- | 
|  | // basic instructions & code generation | 
|  | // ----------------------------------------------------------------------- | 
|  |  | 
|  | // generate the code | 
|  | virtual void reset() = 0; | 
|  | virtual int  generate(const char* name) = 0; | 
|  | virtual void disassemble(const char* name) = 0; | 
|  | virtual int  getCodegenArch() = 0; | 
|  |  | 
|  | // construct prolog and epilog | 
|  | virtual void prolog() = 0; | 
|  | virtual void epilog(uint32_t touched) = 0; | 
|  | virtual void comment(const char* string) = 0; | 
|  |  | 
|  | // data processing... | 
|  | enum { | 
|  | opAND, opEOR, opSUB, opRSB, opADD, opADC, opSBC, opRSC, | 
|  | opTST, opTEQ, opCMP, opCMN, opORR, opMOV, opBIC, opMVN, | 
|  | opADD64, opSUB64 | 
|  | }; | 
|  |  | 
|  | virtual void | 
|  | dataProcessing( int opcode, int cc, int s, | 
|  | int Rd, int Rn, | 
|  | uint32_t Op2) = 0; | 
|  |  | 
|  | // multiply... | 
|  | virtual void MLA(int cc, int s, | 
|  | int Rd, int Rm, int Rs, int Rn) = 0; | 
|  | virtual void MUL(int cc, int s, | 
|  | int Rd, int Rm, int Rs) = 0; | 
|  | virtual void UMULL(int cc, int s, | 
|  | int RdLo, int RdHi, int Rm, int Rs) = 0; | 
|  | virtual void UMUAL(int cc, int s, | 
|  | int RdLo, int RdHi, int Rm, int Rs) = 0; | 
|  | virtual void SMULL(int cc, int s, | 
|  | int RdLo, int RdHi, int Rm, int Rs) = 0; | 
|  | virtual void SMUAL(int cc, int s, | 
|  | int RdLo, int RdHi, int Rm, int Rs) = 0; | 
|  |  | 
|  | // branches... | 
|  | virtual void B(int cc, uint32_t* pc) = 0; | 
|  | virtual void BL(int cc, uint32_t* pc) = 0; | 
|  | virtual void BX(int cc, int Rn) = 0; | 
|  |  | 
|  | virtual void label(const char* theLabel) = 0; | 
|  | virtual void B(int cc, const char* label) = 0; | 
|  | virtual void BL(int cc, const char* label) = 0; | 
|  |  | 
|  | // valid only after generate() has been called | 
|  | virtual uint32_t* pcForLabel(const char* label) = 0; | 
|  |  | 
|  | // data transfer... | 
|  | virtual void LDR (int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed12_pre(0)) = 0; | 
|  | virtual void LDRB(int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed12_pre(0)) = 0; | 
|  | virtual void STR (int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed12_pre(0)) = 0; | 
|  | virtual void STRB(int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed12_pre(0)) = 0; | 
|  |  | 
|  | virtual void LDRH (int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed8_pre(0)) = 0; | 
|  | virtual void LDRSB(int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed8_pre(0)) = 0; | 
|  | virtual void LDRSH(int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed8_pre(0)) = 0; | 
|  | virtual void STRH (int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed8_pre(0)) = 0; | 
|  |  | 
|  | // block data transfer... | 
|  | virtual void LDM(int cc, int dir, | 
|  | int Rn, int W, uint32_t reg_list) = 0; | 
|  | virtual void STM(int cc, int dir, | 
|  | int Rn, int W, uint32_t reg_list) = 0; | 
|  |  | 
|  | // special... | 
|  | virtual void SWP(int cc, int Rn, int Rd, int Rm) = 0; | 
|  | virtual void SWPB(int cc, int Rn, int Rd, int Rm) = 0; | 
|  | virtual void SWI(int cc, uint32_t comment) = 0; | 
|  |  | 
|  | // DSP instructions... | 
|  | enum { | 
|  | // B=0, T=1 | 
|  | //     yx | 
|  | xyBB = 0, // 0000, | 
|  | xyTB = 2, // 0010, | 
|  | xyBT = 4, // 0100, | 
|  | xyTT = 6, // 0110, | 
|  | yB   = 0, // 0000, | 
|  | yT   = 4, // 0100 | 
|  | }; | 
|  |  | 
|  | virtual void PLD(int Rn, uint32_t offset) = 0; | 
|  |  | 
|  | virtual void CLZ(int cc, int Rd, int Rm) = 0; | 
|  |  | 
|  | virtual void QADD(int cc, int Rd, int Rm, int Rn) = 0; | 
|  | virtual void QDADD(int cc, int Rd, int Rm, int Rn) = 0; | 
|  | virtual void QSUB(int cc, int Rd, int Rm, int Rn) = 0; | 
|  | virtual void QDSUB(int cc, int Rd, int Rm, int Rn) = 0; | 
|  |  | 
|  | virtual void SMUL(int cc, int xy, | 
|  | int Rd, int Rm, int Rs) = 0; | 
|  | virtual void SMULW(int cc, int y, | 
|  | int Rd, int Rm, int Rs) = 0; | 
|  | virtual void SMLA(int cc, int xy, | 
|  | int Rd, int Rm, int Rs, int Rn) = 0; | 
|  | virtual void SMLAL(int cc, int xy, | 
|  | int RdHi, int RdLo, int Rs, int Rm) = 0; | 
|  | virtual void SMLAW(int cc, int y, | 
|  | int Rd, int Rm, int Rs, int Rn) = 0; | 
|  |  | 
|  | // byte/half word extract... | 
|  | virtual void UXTB16(int cc, int Rd, int Rm, int rotate) = 0; | 
|  |  | 
|  | // bit manipulation... | 
|  | virtual void UBFX(int cc, int Rd, int Rn, int lsb, int width) = 0; | 
|  |  | 
|  | // ----------------------------------------------------------------------- | 
|  | // convenience... | 
|  | // ----------------------------------------------------------------------- | 
|  | inline void | 
|  | ADC(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opADC, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | ADD(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opADD, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | AND(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opAND, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | BIC(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opBIC, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | EOR(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opEOR, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | MOV(int cc, int s, int Rd, uint32_t Op2) { | 
|  | dataProcessing(opMOV, cc, s, Rd, 0, Op2); | 
|  | } | 
|  | inline void | 
|  | MVN(int cc, int s, int Rd, uint32_t Op2) { | 
|  | dataProcessing(opMVN, cc, s, Rd, 0, Op2); | 
|  | } | 
|  | inline void | 
|  | ORR(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opORR, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | RSB(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opRSB, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | RSC(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opRSC, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | SBC(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opSBC, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | SUB(int cc, int s, int Rd, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opSUB, cc, s, Rd, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | TEQ(int cc, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opTEQ, cc, 1, 0, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | TST(int cc, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opTST, cc, 1, 0, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | CMP(int cc, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opCMP, cc, 1, 0, Rn, Op2); | 
|  | } | 
|  | inline void | 
|  | CMN(int cc, int Rn, uint32_t Op2) { | 
|  | dataProcessing(opCMN, cc, 1, 0, Rn, Op2); | 
|  | } | 
|  |  | 
|  | inline void SMULBB(int cc, int Rd, int Rm, int Rs) { | 
|  | SMUL(cc, xyBB, Rd, Rm, Rs);    } | 
|  | inline void SMULTB(int cc, int Rd, int Rm, int Rs) { | 
|  | SMUL(cc, xyTB, Rd, Rm, Rs);    } | 
|  | inline void SMULBT(int cc, int Rd, int Rm, int Rs) { | 
|  | SMUL(cc, xyBT, Rd, Rm, Rs);    } | 
|  | inline void SMULTT(int cc, int Rd, int Rm, int Rs) { | 
|  | SMUL(cc, xyTT, Rd, Rm, Rs);    } | 
|  |  | 
|  | inline void SMULWB(int cc, int Rd, int Rm, int Rs) { | 
|  | SMULW(cc, yB, Rd, Rm, Rs);    } | 
|  | inline void SMULWT(int cc, int Rd, int Rm, int Rs) { | 
|  | SMULW(cc, yT, Rd, Rm, Rs);    } | 
|  |  | 
|  | inline void | 
|  | SMLABB(int cc, int Rd, int Rm, int Rs, int Rn) { | 
|  | SMLA(cc, xyBB, Rd, Rm, Rs, Rn);    } | 
|  | inline void | 
|  | SMLATB(int cc, int Rd, int Rm, int Rs, int Rn) { | 
|  | SMLA(cc, xyTB, Rd, Rm, Rs, Rn);    } | 
|  | inline void | 
|  | SMLABT(int cc, int Rd, int Rm, int Rs, int Rn) { | 
|  | SMLA(cc, xyBT, Rd, Rm, Rs, Rn);    } | 
|  | inline void | 
|  | SMLATT(int cc, int Rd, int Rm, int Rs, int Rn) { | 
|  | SMLA(cc, xyTT, Rd, Rm, Rs, Rn);    } | 
|  |  | 
|  | inline void | 
|  | SMLALBB(int cc, int RdHi, int RdLo, int Rs, int Rm) { | 
|  | SMLAL(cc, xyBB, RdHi, RdLo, Rs, Rm);    } | 
|  | inline void | 
|  | SMLALTB(int cc, int RdHi, int RdLo, int Rs, int Rm) { | 
|  | SMLAL(cc, xyTB, RdHi, RdLo, Rs, Rm);    } | 
|  | inline void | 
|  | SMLALBT(int cc, int RdHi, int RdLo, int Rs, int Rm) { | 
|  | SMLAL(cc, xyBT, RdHi, RdLo, Rs, Rm);    } | 
|  | inline void | 
|  | SMLALTT(int cc, int RdHi, int RdLo, int Rs, int Rm) { | 
|  | SMLAL(cc, xyTT, RdHi, RdLo, Rs, Rm);    } | 
|  |  | 
|  | inline void | 
|  | SMLAWB(int cc, int Rd, int Rm, int Rs, int Rn) { | 
|  | SMLAW(cc, yB, Rd, Rm, Rs, Rn);    } | 
|  | inline void | 
|  | SMLAWT(int cc, int Rd, int Rm, int Rs, int Rn) { | 
|  | SMLAW(cc, yT, Rd, Rm, Rs, Rn);    } | 
|  |  | 
|  | // Address loading/storing/manipulation | 
|  | virtual void ADDR_LDR(int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed12_pre(0)); | 
|  | virtual void ADDR_STR (int cc, int Rd, | 
|  | int Rn, uint32_t offset = __immed12_pre(0)); | 
|  | virtual void ADDR_ADD(int cc, int s, int Rd, | 
|  | int Rn, uint32_t Op2); | 
|  | virtual void ADDR_SUB(int cc, int s, int Rd, | 
|  | int Rn, uint32_t Op2); | 
|  | }; | 
|  |  | 
|  | }; // namespace android | 
|  |  | 
|  | #endif //ANDROID_ARMASSEMBLER_INTERFACE_H |