1/* libs/pixelflinger/codeflinger/ARMAssemblerInterface.cpp 2** 3** Copyright 2006, The Android Open Source Project 4** 5** Licensed under the Apache License, Version 2.0 (the "License"); 6** you may not use this file except in compliance with the License. 7** You may obtain a copy of the License at 8** 9** http://www.apache.org/licenses/LICENSE-2.0 10** 11** Unless required by applicable law or agreed to in writing, software 12** distributed under the License is distributed on an "AS IS" BASIS, 13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14** See the License for the specific language governing permissions and 15** limitations under the License. 16*/ 17 18 19#include <errno.h> 20#include <stdlib.h> 21#include <stdint.h> 22#include <sys/types.h> 23 24#include <cutils/log.h> 25#include "codeflinger/ARMAssemblerInterface.h" 26 27namespace android { 28 29// ---------------------------------------------------------------------------- 30 31ARMAssemblerInterface::~ARMAssemblerInterface() 32{ 33} 34 35int ARMAssemblerInterface::buildImmediate( 36 uint32_t immediate, uint32_t& rot, uint32_t& imm) 37{ 38 rot = 0; 39 imm = immediate; 40 if (imm > 0x7F) { // skip the easy cases 41 while (!(imm&3) || (imm&0xFC000000)) { 42 uint32_t newval; 43 newval = imm >> 2; 44 newval |= (imm&3) << 30; 45 imm = newval; 46 rot += 2; 47 if (rot == 32) { 48 rot = 0; 49 break; 50 } 51 } 52 } 53 rot = (16 - (rot>>1)) & 0xF; 54 55 if (imm>=0x100) 56 return -EINVAL; 57 58 if (((imm>>(rot<<1)) | (imm<<(32-(rot<<1)))) != immediate) 59 return -1; 60 61 return 0; 62} 63 64// shifters... 65 66bool ARMAssemblerInterface::isValidImmediate(uint32_t immediate) 67{ 68 uint32_t rot, imm; 69 return buildImmediate(immediate, rot, imm) == 0; 70} 71 72uint32_t ARMAssemblerInterface::imm(uint32_t immediate) 73{ 74 uint32_t rot, imm; 75 int err = buildImmediate(immediate, rot, imm); 76 77 LOG_ALWAYS_FATAL_IF(err==-EINVAL, 78 "immediate %08x cannot be encoded", 79 immediate); 80 81 LOG_ALWAYS_FATAL_IF(err, 82 "immediate (%08x) encoding bogus!", 83 immediate); 84 85 return (1<<25) | (rot<<8) | imm; 86} 87 88uint32_t ARMAssemblerInterface::reg_imm(int Rm, int type, uint32_t shift) 89{ 90 return ((shift&0x1F)<<7) | ((type&0x3)<<5) | (Rm&0xF); 91} 92 93uint32_t ARMAssemblerInterface::reg_rrx(int Rm) 94{ 95 return (ROR<<5) | (Rm&0xF); 96} 97 98uint32_t ARMAssemblerInterface::reg_reg(int Rm, int type, int Rs) 99{ 100 return ((Rs&0xF)<<8) | ((type&0x3)<<5) | (1<<4) | (Rm&0xF); 101} 102 103// addressing modes... 104// LDR(B)/STR(B)/PLD (immediate and Rm can be negative, which indicate U=0) 105uint32_t ARMAssemblerInterface::immed12_pre(int32_t immed12, int W) 106{ 107 LOG_ALWAYS_FATAL_IF(abs(immed12) >= 0x800, 108 "LDR(B)/STR(B)/PLD immediate too big (%08x)", 109 immed12); 110 return (1<<24) | (((uint32_t(immed12)>>31)^1)<<23) | 111 ((W&1)<<21) | (abs(immed12)&0x7FF); 112} 113 114uint32_t ARMAssemblerInterface::immed12_post(int32_t immed12) 115{ 116 LOG_ALWAYS_FATAL_IF(abs(immed12) >= 0x800, 117 "LDR(B)/STR(B)/PLD immediate too big (%08x)", 118 immed12); 119 120 return (((uint32_t(immed12)>>31)^1)<<23) | (abs(immed12)&0x7FF); 121} 122 123uint32_t ARMAssemblerInterface::reg_scale_pre(int Rm, int type, 124 uint32_t shift, int W) 125{ 126 return (1<<25) | (1<<24) | 127 (((uint32_t(Rm)>>31)^1)<<23) | ((W&1)<<21) | 128 reg_imm(abs(Rm), type, shift); 129} 130 131uint32_t ARMAssemblerInterface::reg_scale_post(int Rm, int type, uint32_t shift) 132{ 133 return (1<<25) | (((uint32_t(Rm)>>31)^1)<<23) | reg_imm(abs(Rm), type, shift); 134} 135 136// LDRH/LDRSB/LDRSH/STRH (immediate and Rm can be negative, which indicate U=0) 137uint32_t ARMAssemblerInterface::immed8_pre(int32_t immed8, int W) 138{ 139 uint32_t offset = abs(immed8); 140 141 LOG_ALWAYS_FATAL_IF(abs(immed8) >= 0x100, 142 "LDRH/LDRSB/LDRSH/STRH immediate too big (%08x)", 143 immed8); 144 145 return (1<<24) | (1<<22) | (((uint32_t(immed8)>>31)^1)<<23) | 146 ((W&1)<<21) | (((offset&0xF0)<<4)|(offset&0xF)); 147} 148 149uint32_t ARMAssemblerInterface::immed8_post(int32_t immed8) 150{ 151 uint32_t offset = abs(immed8); 152 153 LOG_ALWAYS_FATAL_IF(abs(immed8) >= 0x100, 154 "LDRH/LDRSB/LDRSH/STRH immediate too big (%08x)", 155 immed8); 156 157 return (1<<22) | (((uint32_t(immed8)>>31)^1)<<23) | 158 (((offset&0xF0)<<4) | (offset&0xF)); 159} 160 161uint32_t ARMAssemblerInterface::reg_pre(int Rm, int W) 162{ 163 return (1<<24) | (((uint32_t(Rm)>>31)^1)<<23) | ((W&1)<<21) | (abs(Rm)&0xF); 164} 165 166uint32_t ARMAssemblerInterface::reg_post(int Rm) 167{ 168 return (((uint32_t(Rm)>>31)^1)<<23) | (abs(Rm)&0xF); 169} 170 171 172}; // namespace android 173 174