説明を見る。00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037 #ifndef V8_ASSEMBLER_IA32_INL_H_
00038 #define V8_ASSEMBLER_IA32_INL_H_
00039
00040 #include "cpu.h"
00041
00042 namespace v8 { namespace internal {
00043
00044 Condition NegateCondition(Condition cc) {
00045 return static_cast<Condition>(cc ^ 1);
00046 }
00047
00048
00049
00050 void RelocInfo::apply(int delta) {
00051 if (rmode_ == RUNTIME_ENTRY || IsCodeTarget(rmode_)) {
00052 int32_t* p = reinterpret_cast<int32_t*>(pc_);
00053 *p -= delta;
00054 } else if (rmode_ == JS_RETURN && is_call_instruction()) {
00055
00056
00057 int32_t* p = reinterpret_cast<int32_t*>(pc_ + 1);
00058 *p -= delta;
00059 } else if (IsInternalReference(rmode_)) {
00060
00061 int32_t* p = reinterpret_cast<int32_t*>(pc_);
00062 *p += delta;
00063 }
00064 }
00065
00066
00067 Address RelocInfo::target_address() {
00068 ASSERT(IsCodeTarget(rmode_) || rmode_ == RUNTIME_ENTRY);
00069 return Assembler::target_address_at(pc_);
00070 }
00071
00072
00073 void RelocInfo::set_target_address(Address target) {
00074 ASSERT(IsCodeTarget(rmode_) || rmode_ == RUNTIME_ENTRY);
00075 Assembler::set_target_address_at(pc_, target);
00076 }
00077
00078
00079 Object* RelocInfo::target_object() {
00080 ASSERT(IsCodeTarget(rmode_) || rmode_ == EMBEDDED_OBJECT);
00081 return *reinterpret_cast<Object**>(pc_);
00082 }
00083
00084
00085 Object** RelocInfo::target_object_address() {
00086 ASSERT(IsCodeTarget(rmode_) || rmode_ == EMBEDDED_OBJECT);
00087 return reinterpret_cast<Object**>(pc_);
00088 }
00089
00090
00091 void RelocInfo::set_target_object(Object* target) {
00092 ASSERT(IsCodeTarget(rmode_) || rmode_ == EMBEDDED_OBJECT);
00093 *reinterpret_cast<Object**>(pc_) = target;
00094 }
00095
00096
00097 Address* RelocInfo::target_reference_address() {
00098 ASSERT(rmode_ == RelocInfo::EXTERNAL_REFERENCE);
00099 return reinterpret_cast<Address*>(pc_);
00100 }
00101
00102
00103 Address RelocInfo::call_address() {
00104 ASSERT(is_call_instruction());
00105 return Assembler::target_address_at(pc_ + 1);
00106 }
00107
00108
00109 void RelocInfo::set_call_address(Address target) {
00110 ASSERT(is_call_instruction());
00111 Assembler::set_target_address_at(pc_ + 1, target);
00112 }
00113
00114
00115 Object* RelocInfo::call_object() {
00116 ASSERT(is_call_instruction());
00117 return *call_object_address();
00118 }
00119
00120
00121 Object** RelocInfo::call_object_address() {
00122 ASSERT(is_call_instruction());
00123 return reinterpret_cast<Object**>(pc_ + 1);
00124 }
00125
00126
00127 void RelocInfo::set_call_object(Object* target) {
00128 ASSERT(is_call_instruction());
00129 *call_object_address() = target;
00130 }
00131
00132
00133 bool RelocInfo::is_call_instruction() {
00134 return *pc_ == 0xE8;
00135 }
00136
00137
00138 Immediate::Immediate(int x) {
00139 x_ = x;
00140 rmode_ = RelocInfo::NONE;
00141 }
00142
00143
00144 Immediate::Immediate(const ExternalReference& ext) {
00145 x_ = reinterpret_cast<int32_t>(ext.address());
00146 rmode_ = RelocInfo::EXTERNAL_REFERENCE;
00147 }
00148
00149 Immediate::Immediate(const char* s) {
00150 x_ = reinterpret_cast<int32_t>(s);
00151 rmode_ = RelocInfo::EMBEDDED_STRING;
00152 }
00153
00154
00155 Immediate::Immediate(Handle<Object> handle) {
00156
00157 Object* obj = *handle;
00158 ASSERT(!Heap::InNewSpace(obj));
00159 if (obj->IsHeapObject()) {
00160 x_ = reinterpret_cast<intptr_t>(handle.location());
00161 rmode_ = RelocInfo::EMBEDDED_OBJECT;
00162 } else {
00163
00164 x_ = reinterpret_cast<intptr_t>(obj);
00165 rmode_ = RelocInfo::NONE;
00166 }
00167 }
00168
00169
00170 Immediate::Immediate(Smi* value) {
00171 x_ = reinterpret_cast<intptr_t>(value);
00172 rmode_ = RelocInfo::NONE;
00173 }
00174
00175
00176 void Assembler::emit(uint32_t x) {
00177 *reinterpret_cast<uint32_t*>(pc_) = x;
00178 pc_ += sizeof(uint32_t);
00179 }
00180
00181
00182 void Assembler::emit(Handle<Object> handle) {
00183
00184 Object* obj = *handle;
00185 ASSERT(!Heap::InNewSpace(obj));
00186 if (obj->IsHeapObject()) {
00187 emit(reinterpret_cast<intptr_t>(handle.location()),
00188 RelocInfo::EMBEDDED_OBJECT);
00189 } else {
00190
00191 emit(reinterpret_cast<intptr_t>(obj));
00192 }
00193 }
00194
00195
00196 void Assembler::emit(uint32_t x, RelocInfo::Mode rmode) {
00197 if (rmode != RelocInfo::NONE) RecordRelocInfo(rmode);
00198 emit(x);
00199 }
00200
00201
00202 void Assembler::emit(const Immediate& x) {
00203 if (x.rmode_ != RelocInfo::NONE) RecordRelocInfo(x.rmode_);
00204 emit(x.x_);
00205 }
00206
00207
00208 Address Assembler::target_address_at(Address pc) {
00209 return pc + sizeof(int32_t) + *reinterpret_cast<int32_t*>(pc);
00210 }
00211
00212
00213 void Assembler::set_target_address_at(Address pc, Address target) {
00214 int32_t* p = reinterpret_cast<int32_t*>(pc);
00215 *p = target - (pc + sizeof(int32_t));
00216 CPU::FlushICache(p, sizeof(int32_t));
00217 }
00218
00219
00220 Displacement Assembler::disp_at(Label* L) {
00221 return Displacement(long_at(L->pos()));
00222 }
00223
00224
00225 void Assembler::disp_at_put(Label* L, Displacement disp) {
00226 long_at_put(L->pos(), disp.data());
00227 }
00228
00229
00230 void Assembler::emit_disp(Label* L, Displacement::Type type) {
00231 Displacement disp(L, type);
00232 L->link_to(pc_offset());
00233 emit(static_cast<int>(disp.data()));
00234 }
00235
00236
00237 void Operand::set_modrm(int mod,
00238 Register rm) {
00239 ASSERT((mod & -4) == 0);
00240 buf_[0] = mod << 6 | rm.code();
00241 len_ = 1;
00242 }
00243
00244
00245 void Operand::set_dispr(int32_t disp, RelocInfo::Mode rmode) {
00246 ASSERT(len_ == 1 || len_ == 2);
00247 *reinterpret_cast<int32_t*>(&buf_[len_]) = disp;
00248 len_ += sizeof(int32_t);
00249 rmode_ = rmode;
00250 }
00251
00252 Operand::Operand(Register reg) {
00253
00254 set_modrm(3, reg);
00255 }
00256
00257
00258 Operand::Operand(int32_t disp, RelocInfo::Mode rmode) {
00259
00260 set_modrm(0, ebp);
00261 set_dispr(disp, rmode);
00262 }
00263
00264 } }
00265
00266 #endif // V8_ASSEMBLER_IA32_INL_H_