Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(2)

Side by Side Diff: src/ia32/macro-assembler-ia32.h

Issue 1463803002: [debugger] flood function for stepping before calling it. (Closed) Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: rebase, ports, deoptimize builtins Created 5 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #ifndef V8_IA32_MACRO_ASSEMBLER_IA32_H_ 5 #ifndef V8_IA32_MACRO_ASSEMBLER_IA32_H_
6 #define V8_IA32_MACRO_ASSEMBLER_IA32_H_ 6 #define V8_IA32_MACRO_ASSEMBLER_IA32_H_
7 7
8 #include "src/assembler.h" 8 #include "src/assembler.h"
9 #include "src/bailout-reason.h" 9 #include "src/bailout-reason.h"
10 #include "src/frames.h" 10 #include "src/frames.h"
(...skipping 23 matching lines...) Expand all
34 // distinguish memory operands from other operands on ia32. 34 // distinguish memory operands from other operands on ia32.
35 typedef Operand MemOperand; 35 typedef Operand MemOperand;
36 36
37 enum RememberedSetAction { EMIT_REMEMBERED_SET, OMIT_REMEMBERED_SET }; 37 enum RememberedSetAction { EMIT_REMEMBERED_SET, OMIT_REMEMBERED_SET };
38 enum SmiCheck { INLINE_SMI_CHECK, OMIT_SMI_CHECK }; 38 enum SmiCheck { INLINE_SMI_CHECK, OMIT_SMI_CHECK };
39 enum PointersToHereCheck { 39 enum PointersToHereCheck {
40 kPointersToHereMaybeInteresting, 40 kPointersToHereMaybeInteresting,
41 kPointersToHereAreAlwaysInteresting 41 kPointersToHereAreAlwaysInteresting
42 }; 42 };
43 43
44 44 enum RegisterValueType { REGISTER_VALUE_IS_SMI, REGISTER_VALUE_IS_INT32 };
45 enum RegisterValueType {
46 REGISTER_VALUE_IS_SMI,
47 REGISTER_VALUE_IS_INT32
48 };
49
50 45
51 #ifdef DEBUG 46 #ifdef DEBUG
52 bool AreAliased(Register reg1, 47 bool AreAliased(Register reg1, Register reg2, Register reg3 = no_reg,
53 Register reg2, 48 Register reg4 = no_reg, Register reg5 = no_reg,
54 Register reg3 = no_reg, 49 Register reg6 = no_reg, Register reg7 = no_reg,
55 Register reg4 = no_reg,
56 Register reg5 = no_reg,
57 Register reg6 = no_reg,
58 Register reg7 = no_reg,
59 Register reg8 = no_reg); 50 Register reg8 = no_reg);
60 #endif 51 #endif
61 52
62
63 // MacroAssembler implements a collection of frequently used macros. 53 // MacroAssembler implements a collection of frequently used macros.
64 class MacroAssembler: public Assembler { 54 class MacroAssembler: public Assembler {
65 public: 55 public:
66 // The isolate parameter can be NULL if the macro assembler should 56 // The isolate parameter can be NULL if the macro assembler should
67 // not use isolate-dependent functionality. In this case, it's the 57 // not use isolate-dependent functionality. In this case, it's the
68 // responsibility of the caller to never invoke such function on the 58 // responsibility of the caller to never invoke such function on the
69 // macro assembler. 59 // macro assembler.
70 MacroAssembler(Isolate* isolate, void* buffer, int size); 60 MacroAssembler(Isolate* isolate, void* buffer, int size);
71 61
72 void Load(Register dst, const Operand& src, Representation r); 62 void Load(Register dst, const Operand& src, Representation r);
(...skipping 29 matching lines...) Expand all
102 // Compare the object in a register to a value and jump if they are not equal. 92 // Compare the object in a register to a value and jump if they are not equal.
103 void JumpIfNotRoot(Register with, Heap::RootListIndex index, 93 void JumpIfNotRoot(Register with, Heap::RootListIndex index,
104 Label* if_not_equal, 94 Label* if_not_equal,
105 Label::Distance if_not_equal_distance = Label::kNear) { 95 Label::Distance if_not_equal_distance = Label::kNear) {
106 CompareRoot(with, index); 96 CompareRoot(with, index);
107 j(not_equal, if_not_equal, if_not_equal_distance); 97 j(not_equal, if_not_equal, if_not_equal_distance);
108 } 98 }
109 99
110 // --------------------------------------------------------------------------- 100 // ---------------------------------------------------------------------------
111 // GC Support 101 // GC Support
112 enum RememberedSetFinalAction { 102 enum RememberedSetFinalAction { kReturnAtEnd, kFallThroughAtEnd };
113 kReturnAtEnd,
114 kFallThroughAtEnd
115 };
116 103
117 // Record in the remembered set the fact that we have a pointer to new space 104 // Record in the remembered set the fact that we have a pointer to new space
118 // at the address pointed to by the addr register. Only works if addr is not 105 // at the address pointed to by the addr register. Only works if addr is not
119 // in new space. 106 // in new space.
120 void RememberedSetHelper(Register object, // Used for debug code. 107 void RememberedSetHelper(Register object, // Used for debug code.
121 Register addr, 108 Register addr, Register scratch,
122 Register scratch,
123 SaveFPRegsMode save_fp, 109 SaveFPRegsMode save_fp,
124 RememberedSetFinalAction and_then); 110 RememberedSetFinalAction and_then);
125 111
126 void CheckPageFlag(Register object, 112 void CheckPageFlag(Register object, Register scratch, int mask, Condition cc,
127 Register scratch,
128 int mask,
129 Condition cc,
130 Label* condition_met, 113 Label* condition_met,
131 Label::Distance condition_met_distance = Label::kFar); 114 Label::Distance condition_met_distance = Label::kFar);
132 115
133 void CheckPageFlagForMap( 116 void CheckPageFlagForMap(
134 Handle<Map> map, 117 Handle<Map> map, int mask, Condition cc, Label* condition_met,
135 int mask,
136 Condition cc,
137 Label* condition_met,
138 Label::Distance condition_met_distance = Label::kFar); 118 Label::Distance condition_met_distance = Label::kFar);
139 119
140 // Check if object is in new space. Jumps if the object is not in new space. 120 // Check if object is in new space. Jumps if the object is not in new space.
141 // The register scratch can be object itself, but scratch will be clobbered. 121 // The register scratch can be object itself, but scratch will be clobbered.
142 void JumpIfNotInNewSpace(Register object, 122 void JumpIfNotInNewSpace(Register object, Register scratch, Label* branch,
143 Register scratch,
144 Label* branch,
145 Label::Distance distance = Label::kFar) { 123 Label::Distance distance = Label::kFar) {
146 InNewSpace(object, scratch, zero, branch, distance); 124 InNewSpace(object, scratch, zero, branch, distance);
147 } 125 }
148 126
149 // Check if object is in new space. Jumps if the object is in new space. 127 // Check if object is in new space. Jumps if the object is in new space.
150 // The register scratch can be object itself, but it will be clobbered. 128 // The register scratch can be object itself, but it will be clobbered.
151 void JumpIfInNewSpace(Register object, 129 void JumpIfInNewSpace(Register object, Register scratch, Label* branch,
152 Register scratch,
153 Label* branch,
154 Label::Distance distance = Label::kFar) { 130 Label::Distance distance = Label::kFar) {
155 InNewSpace(object, scratch, not_zero, branch, distance); 131 InNewSpace(object, scratch, not_zero, branch, distance);
156 } 132 }
157 133
158 // Check if an object has a given incremental marking color. Also uses ecx! 134 // Check if an object has a given incremental marking color. Also uses ecx!
159 void HasColor(Register object, 135 void HasColor(Register object, Register scratch0, Register scratch1,
160 Register scratch0, 136 Label* has_color, Label::Distance has_color_distance,
161 Register scratch1, 137 int first_bit, int second_bit);
162 Label* has_color,
163 Label::Distance has_color_distance,
164 int first_bit,
165 int second_bit);
166 138
167 void JumpIfBlack(Register object, 139 void JumpIfBlack(Register object, Register scratch0, Register scratch1,
168 Register scratch0,
169 Register scratch1,
170 Label* on_black, 140 Label* on_black,
171 Label::Distance on_black_distance = Label::kFar); 141 Label::Distance on_black_distance = Label::kFar);
172 142
173 // Checks the color of an object. If the object is already grey or black 143 // Checks the color of an object. If the object is already grey or black
174 // then we just fall through, since it is already live. If it is white and 144 // then we just fall through, since it is already live. If it is white and
175 // we can determine that it doesn't need to be scanned, then we just mark it 145 // we can determine that it doesn't need to be scanned, then we just mark it
176 // black and fall through. For the rest we jump to the label so the 146 // black and fall through. For the rest we jump to the label so the
177 // incremental marker can fix its assumptions. 147 // incremental marker can fix its assumptions.
178 void EnsureNotWhite(Register object, 148 void EnsureNotWhite(Register object, Register scratch1, Register scratch2,
179 Register scratch1,
180 Register scratch2,
181 Label* object_is_white_and_not_data, 149 Label* object_is_white_and_not_data,
182 Label::Distance distance); 150 Label::Distance distance);
183 151
184 // Notify the garbage collector that we wrote a pointer into an object. 152 // Notify the garbage collector that we wrote a pointer into an object.
185 // |object| is the object being stored into, |value| is the object being 153 // |object| is the object being stored into, |value| is the object being
186 // stored. value and scratch registers are clobbered by the operation. 154 // stored. value and scratch registers are clobbered by the operation.
187 // The offset is the offset from the start of the object, not the offset from 155 // The offset is the offset from the start of the object, not the offset from
188 // the tagged HeapObject pointer. For use with FieldOperand(reg, off). 156 // the tagged HeapObject pointer. For use with FieldOperand(reg, off).
189 void RecordWriteField( 157 void RecordWriteField(
190 Register object, 158 Register object, int offset, Register value, Register scratch,
191 int offset,
192 Register value,
193 Register scratch,
194 SaveFPRegsMode save_fp, 159 SaveFPRegsMode save_fp,
195 RememberedSetAction remembered_set_action = EMIT_REMEMBERED_SET, 160 RememberedSetAction remembered_set_action = EMIT_REMEMBERED_SET,
196 SmiCheck smi_check = INLINE_SMI_CHECK, 161 SmiCheck smi_check = INLINE_SMI_CHECK,
197 PointersToHereCheck pointers_to_here_check_for_value = 162 PointersToHereCheck pointers_to_here_check_for_value =
198 kPointersToHereMaybeInteresting); 163 kPointersToHereMaybeInteresting);
199 164
200 // As above, but the offset has the tag presubtracted. For use with 165 // As above, but the offset has the tag presubtracted. For use with
201 // Operand(reg, off). 166 // Operand(reg, off).
202 void RecordWriteContextSlot( 167 void RecordWriteContextSlot(
203 Register context, 168 Register context, int offset, Register value, Register scratch,
204 int offset,
205 Register value,
206 Register scratch,
207 SaveFPRegsMode save_fp, 169 SaveFPRegsMode save_fp,
208 RememberedSetAction remembered_set_action = EMIT_REMEMBERED_SET, 170 RememberedSetAction remembered_set_action = EMIT_REMEMBERED_SET,
209 SmiCheck smi_check = INLINE_SMI_CHECK, 171 SmiCheck smi_check = INLINE_SMI_CHECK,
210 PointersToHereCheck pointers_to_here_check_for_value = 172 PointersToHereCheck pointers_to_here_check_for_value =
211 kPointersToHereMaybeInteresting) { 173 kPointersToHereMaybeInteresting) {
212 RecordWriteField(context, 174 RecordWriteField(context, offset + kHeapObjectTag, value, scratch, save_fp,
213 offset + kHeapObjectTag, 175 remembered_set_action, smi_check,
214 value,
215 scratch,
216 save_fp,
217 remembered_set_action,
218 smi_check,
219 pointers_to_here_check_for_value); 176 pointers_to_here_check_for_value);
220 } 177 }
221 178
222 // Notify the garbage collector that we wrote a pointer into a fixed array. 179 // Notify the garbage collector that we wrote a pointer into a fixed array.
223 // |array| is the array being stored into, |value| is the 180 // |array| is the array being stored into, |value| is the
224 // object being stored. |index| is the array index represented as a 181 // object being stored. |index| is the array index represented as a
225 // Smi. All registers are clobbered by the operation RecordWriteArray 182 // Smi. All registers are clobbered by the operation RecordWriteArray
226 // filters out smis so it does not update the write barrier if the 183 // filters out smis so it does not update the write barrier if the
227 // value is a smi. 184 // value is a smi.
228 void RecordWriteArray( 185 void RecordWriteArray(
229 Register array, 186 Register array, Register value, Register index, SaveFPRegsMode save_fp,
230 Register value,
231 Register index,
232 SaveFPRegsMode save_fp,
233 RememberedSetAction remembered_set_action = EMIT_REMEMBERED_SET, 187 RememberedSetAction remembered_set_action = EMIT_REMEMBERED_SET,
234 SmiCheck smi_check = INLINE_SMI_CHECK, 188 SmiCheck smi_check = INLINE_SMI_CHECK,
235 PointersToHereCheck pointers_to_here_check_for_value = 189 PointersToHereCheck pointers_to_here_check_for_value =
236 kPointersToHereMaybeInteresting); 190 kPointersToHereMaybeInteresting);
237 191
238 // For page containing |object| mark region covering |address| 192 // For page containing |object| mark region covering |address|
239 // dirty. |object| is the object being stored into, |value| is the 193 // dirty. |object| is the object being stored into, |value| is the
240 // object being stored. The address and value registers are clobbered by the 194 // object being stored. The address and value registers are clobbered by the
241 // operation. RecordWrite filters out smis so it does not update the 195 // operation. RecordWrite filters out smis so it does not update the
242 // write barrier if the value is a smi. 196 // write barrier if the value is a smi.
243 void RecordWrite( 197 void RecordWrite(
244 Register object, 198 Register object, Register address, Register value, SaveFPRegsMode save_fp,
245 Register address,
246 Register value,
247 SaveFPRegsMode save_fp,
248 RememberedSetAction remembered_set_action = EMIT_REMEMBERED_SET, 199 RememberedSetAction remembered_set_action = EMIT_REMEMBERED_SET,
249 SmiCheck smi_check = INLINE_SMI_CHECK, 200 SmiCheck smi_check = INLINE_SMI_CHECK,
250 PointersToHereCheck pointers_to_here_check_for_value = 201 PointersToHereCheck pointers_to_here_check_for_value =
251 kPointersToHereMaybeInteresting); 202 kPointersToHereMaybeInteresting);
252 203
253 // For page containing |object| mark the region covering the object's map 204 // For page containing |object| mark the region covering the object's map
254 // dirty. |object| is the object being stored into, |map| is the Map object 205 // dirty. |object| is the object being stored into, |map| is the Map object
255 // that was stored. 206 // that was stored.
256 void RecordWriteForMap( 207 void RecordWriteForMap(Register object, Handle<Map> map, Register scratch1,
257 Register object, 208 Register scratch2, SaveFPRegsMode save_fp);
258 Handle<Map> map,
259 Register scratch1,
260 Register scratch2,
261 SaveFPRegsMode save_fp);
262 209
263 // --------------------------------------------------------------------------- 210 // ---------------------------------------------------------------------------
264 // Debugger Support 211 // Debugger Support
265 212
266 void DebugBreak(); 213 void DebugBreak();
267 214
268 // Generates function and stub prologue code. 215 // Generates function and stub prologue code.
269 void StubPrologue(); 216 void StubPrologue();
270 void Prologue(bool code_pre_aging); 217 void Prologue(bool code_pre_aging);
271 218
(...skipping 17 matching lines...) Expand all
289 // Find the function context up the context chain. 236 // Find the function context up the context chain.
290 void LoadContext(Register dst, int context_chain_length); 237 void LoadContext(Register dst, int context_chain_length);
291 238
292 // Load the global proxy from the current context. 239 // Load the global proxy from the current context.
293 void LoadGlobalProxy(Register dst); 240 void LoadGlobalProxy(Register dst);
294 241
295 // Conditionally load the cached Array transitioned map of type 242 // Conditionally load the cached Array transitioned map of type
296 // transitioned_kind from the native context if the map in register 243 // transitioned_kind from the native context if the map in register
297 // map_in_out is the cached Array map in the native context of 244 // map_in_out is the cached Array map in the native context of
298 // expected_kind. 245 // expected_kind.
299 void LoadTransitionedArrayMapConditional( 246 void LoadTransitionedArrayMapConditional(ElementsKind expected_kind,
300 ElementsKind expected_kind, 247 ElementsKind transitioned_kind,
301 ElementsKind transitioned_kind, 248 Register map_in_out,
302 Register map_in_out, 249 Register scratch,
303 Register scratch, 250 Label* no_map_match);
304 Label* no_map_match);
305 251
306 // Load the global function with the given index. 252 // Load the global function with the given index.
307 void LoadGlobalFunction(int index, Register function); 253 void LoadGlobalFunction(int index, Register function);
308 254
309 // Load the initial map from the global function. The registers 255 // Load the initial map from the global function. The registers
310 // function and map can be the same. 256 // function and map can be the same.
311 void LoadGlobalFunctionInitialMap(Register function, Register map); 257 void LoadGlobalFunctionInitialMap(Register function, Register map);
312 258
313 // Push and pop the registers that can hold pointers. 259 // Push and pop the registers that can hold pointers.
314 void PushSafepointRegisters() { pushad(); } 260 void PushSafepointRegisters() { pushad(); }
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after
347 void GetWeakValue(Register value, Handle<WeakCell> cell); 293 void GetWeakValue(Register value, Handle<WeakCell> cell);
348 294
349 // Load the value of the weak cell in the value register. Branch to the given 295 // Load the value of the weak cell in the value register. Branch to the given
350 // miss label if the weak cell was cleared. 296 // miss label if the weak cell was cleared.
351 void LoadWeakValue(Register value, Handle<WeakCell> cell, Label* miss); 297 void LoadWeakValue(Register value, Handle<WeakCell> cell, Label* miss);
352 298
353 // --------------------------------------------------------------------------- 299 // ---------------------------------------------------------------------------
354 // JavaScript invokes 300 // JavaScript invokes
355 301
356 // Invoke the JavaScript function code by either calling or jumping. 302 // Invoke the JavaScript function code by either calling or jumping.
357 void InvokeCode(Register code,
358 const ParameterCount& expected,
359 const ParameterCount& actual,
360 InvokeFlag flag,
361 const CallWrapper& call_wrapper) {
362 InvokeCode(Operand(code), no_reg, expected, actual, flag, call_wrapper);
363 }
364 303
365 void InvokeCode(const Operand& code, 304 void InvokeFunctionCode(Register function, Register new_target,
366 Register new_target, 305 const ParameterCount& expected,
367 const ParameterCount& expected, 306 const ParameterCount& actual, InvokeFlag flag,
368 const ParameterCount& actual, 307 const CallWrapper& call_wrapper);
369 InvokeFlag flag,
370 const CallWrapper& call_wrapper);
371 308
372 // Invoke the JavaScript function in the given register. Changes the 309 // Invoke the JavaScript function in the given register. Changes the
373 // current context to the context in the function before invoking. 310 // current context to the context in the function before invoking.
374 void InvokeFunction(Register function, 311 void InvokeFunction(Register function, Register new_target,
375 Register new_target, 312 const ParameterCount& actual, InvokeFlag flag,
376 const ParameterCount& actual,
377 InvokeFlag flag,
378 const CallWrapper& call_wrapper); 313 const CallWrapper& call_wrapper);
379 314
380 void InvokeFunction(Register function, 315 void InvokeFunction(Register function, const ParameterCount& expected,
381 const ParameterCount& expected, 316 const ParameterCount& actual, InvokeFlag flag,
382 const ParameterCount& actual,
383 InvokeFlag flag,
384 const CallWrapper& call_wrapper); 317 const CallWrapper& call_wrapper);
385 318
386 void InvokeFunction(Handle<JSFunction> function, 319 void InvokeFunction(Handle<JSFunction> function,
387 const ParameterCount& expected, 320 const ParameterCount& expected,
388 const ParameterCount& actual, 321 const ParameterCount& actual, InvokeFlag flag,
389 InvokeFlag flag,
390 const CallWrapper& call_wrapper); 322 const CallWrapper& call_wrapper);
391 323
392 // Invoke specified builtin JavaScript function. 324 // Invoke specified builtin JavaScript function.
393 void InvokeBuiltin(int native_context_index, InvokeFlag flag, 325 void InvokeBuiltin(int native_context_index, InvokeFlag flag,
394 const CallWrapper& call_wrapper = NullCallWrapper()); 326 const CallWrapper& call_wrapper = NullCallWrapper());
395 327
396 // Store the function for the given builtin in the target register. 328 // Store the function for the given builtin in the target register.
397 void GetBuiltinFunction(Register target, int native_context_index); 329 void GetBuiltinFunction(Register target, int native_context_index);
398 330
399 // Store the code object for the given builtin in the target register.
400 void GetBuiltinEntry(Register target, int native_context_index);
401
402 // Expression support 331 // Expression support
403 // cvtsi2sd instruction only writes to the low 64-bit of dst register, which 332 // cvtsi2sd instruction only writes to the low 64-bit of dst register, which
404 // hinders register renaming and makes dependence chains longer. So we use 333 // hinders register renaming and makes dependence chains longer. So we use
405 // xorps to clear the dst register before cvtsi2sd to solve this issue. 334 // xorps to clear the dst register before cvtsi2sd to solve this issue.
406 void Cvtsi2sd(XMMRegister dst, Register src) { Cvtsi2sd(dst, Operand(src)); } 335 void Cvtsi2sd(XMMRegister dst, Register src) { Cvtsi2sd(dst, Operand(src)); }
407 void Cvtsi2sd(XMMRegister dst, const Operand& src); 336 void Cvtsi2sd(XMMRegister dst, const Operand& src);
408 337
409 // Support for constant splitting. 338 // Support for constant splitting.
410 bool IsUnsafeImmediate(const Immediate& x); 339 bool IsUnsafeImmediate(const Immediate& x);
411 void SafeMove(Register dst, const Immediate& x); 340 void SafeMove(Register dst, const Immediate& x);
412 void SafePush(const Immediate& x); 341 void SafePush(const Immediate& x);
413 342
414 // Compare object type for heap object. 343 // Compare object type for heap object.
415 // Incoming register is heap_object and outgoing register is map. 344 // Incoming register is heap_object and outgoing register is map.
416 void CmpObjectType(Register heap_object, InstanceType type, Register map); 345 void CmpObjectType(Register heap_object, InstanceType type, Register map);
417 346
418 // Compare instance type for map. 347 // Compare instance type for map.
419 void CmpInstanceType(Register map, InstanceType type); 348 void CmpInstanceType(Register map, InstanceType type);
420 349
421 // Check if a map for a JSObject indicates that the object has fast elements. 350 // Check if a map for a JSObject indicates that the object has fast elements.
422 // Jump to the specified label if it does not. 351 // Jump to the specified label if it does not.
423 void CheckFastElements(Register map, 352 void CheckFastElements(Register map, Label* fail,
424 Label* fail,
425 Label::Distance distance = Label::kFar); 353 Label::Distance distance = Label::kFar);
426 354
427 // Check if a map for a JSObject indicates that the object can have both smi 355 // Check if a map for a JSObject indicates that the object can have both smi
428 // and HeapObject elements. Jump to the specified label if it does not. 356 // and HeapObject elements. Jump to the specified label if it does not.
429 void CheckFastObjectElements(Register map, 357 void CheckFastObjectElements(Register map, Label* fail,
430 Label* fail,
431 Label::Distance distance = Label::kFar); 358 Label::Distance distance = Label::kFar);
432 359
433 // Check if a map for a JSObject indicates that the object has fast smi only 360 // Check if a map for a JSObject indicates that the object has fast smi only
434 // elements. Jump to the specified label if it does not. 361 // elements. Jump to the specified label if it does not.
435 void CheckFastSmiElements(Register map, 362 void CheckFastSmiElements(Register map, Label* fail,
436 Label* fail,
437 Label::Distance distance = Label::kFar); 363 Label::Distance distance = Label::kFar);
438 364
439 // Check to see if maybe_number can be stored as a double in 365 // Check to see if maybe_number can be stored as a double in
440 // FastDoubleElements. If it can, store it at the index specified by key in 366 // FastDoubleElements. If it can, store it at the index specified by key in
441 // the FastDoubleElements array elements, otherwise jump to fail. 367 // the FastDoubleElements array elements, otherwise jump to fail.
442 void StoreNumberToDoubleElements(Register maybe_number, 368 void StoreNumberToDoubleElements(Register maybe_number, Register elements,
443 Register elements, 369 Register key, Register scratch1,
444 Register key, 370 XMMRegister scratch2, Label* fail,
445 Register scratch1,
446 XMMRegister scratch2,
447 Label* fail,
448 int offset = 0); 371 int offset = 0);
449 372
450 // Compare an object's map with the specified map. 373 // Compare an object's map with the specified map.
451 void CompareMap(Register obj, Handle<Map> map); 374 void CompareMap(Register obj, Handle<Map> map);
452 375
453 // Check if the map of an object is equal to a specified map and branch to 376 // Check if the map of an object is equal to a specified map and branch to
454 // label if not. Skip the smi check if not required (object is known to be a 377 // label if not. Skip the smi check if not required (object is known to be a
455 // heap object). If mode is ALLOW_ELEMENT_TRANSITION_MAPS, then also match 378 // heap object). If mode is ALLOW_ELEMENT_TRANSITION_MAPS, then also match
456 // against maps that are ElementsKind transition maps of the specified map. 379 // against maps that are ElementsKind transition maps of the specified map.
457 void CheckMap(Register obj, 380 void CheckMap(Register obj, Handle<Map> map, Label* fail,
458 Handle<Map> map,
459 Label* fail,
460 SmiCheckType smi_check_type); 381 SmiCheckType smi_check_type);
461 382
462 // Check if the map of an object is equal to a specified weak map and branch 383 // Check if the map of an object is equal to a specified weak map and branch
463 // to a specified target if equal. Skip the smi check if not required 384 // to a specified target if equal. Skip the smi check if not required
464 // (object is known to be a heap object) 385 // (object is known to be a heap object)
465 void DispatchWeakMap(Register obj, Register scratch1, Register scratch2, 386 void DispatchWeakMap(Register obj, Register scratch1, Register scratch2,
466 Handle<WeakCell> cell, Handle<Code> success, 387 Handle<WeakCell> cell, Handle<Code> success,
467 SmiCheckType smi_check_type); 388 SmiCheckType smi_check_type);
468 389
469 // Check if the object in register heap_object is a string. Afterwards the 390 // Check if the object in register heap_object is a string. Afterwards the
470 // register map contains the object map and the register instance_type 391 // register map contains the object map and the register instance_type
471 // contains the instance_type. The registers map and instance_type can be the 392 // contains the instance_type. The registers map and instance_type can be the
472 // same in which case it contains the instance type afterwards. Either of the 393 // same in which case it contains the instance type afterwards. Either of the
473 // registers map and instance_type can be the same as heap_object. 394 // registers map and instance_type can be the same as heap_object.
474 Condition IsObjectStringType(Register heap_object, 395 Condition IsObjectStringType(Register heap_object, Register map,
475 Register map,
476 Register instance_type); 396 Register instance_type);
477 397
478 // Check if the object in register heap_object is a name. Afterwards the 398 // Check if the object in register heap_object is a name. Afterwards the
479 // register map contains the object map and the register instance_type 399 // register map contains the object map and the register instance_type
480 // contains the instance_type. The registers map and instance_type can be the 400 // contains the instance_type. The registers map and instance_type can be the
481 // same in which case it contains the instance type afterwards. Either of the 401 // same in which case it contains the instance type afterwards. Either of the
482 // registers map and instance_type can be the same as heap_object. 402 // registers map and instance_type can be the same as heap_object.
483 Condition IsObjectNameType(Register heap_object, 403 Condition IsObjectNameType(Register heap_object, Register map,
484 Register map,
485 Register instance_type); 404 Register instance_type);
486 405
487 // FCmp is similar to integer cmp, but requires unsigned 406 // FCmp is similar to integer cmp, but requires unsigned
488 // jcc instructions (je, ja, jae, jb, jbe, je, and jz). 407 // jcc instructions (je, ja, jae, jb, jbe, je, and jz).
489 void FCmp(); 408 void FCmp();
490 409
491 void ClampUint8(Register reg); 410 void ClampUint8(Register reg);
492 411
493 void ClampDoubleToUint8(XMMRegister input_reg, 412 void ClampDoubleToUint8(XMMRegister input_reg, XMMRegister scratch_reg,
494 XMMRegister scratch_reg,
495 Register result_reg); 413 Register result_reg);
496 414
497 void SlowTruncateToI(Register result_reg, Register input_reg, 415 void SlowTruncateToI(Register result_reg, Register input_reg,
498 int offset = HeapNumber::kValueOffset - kHeapObjectTag); 416 int offset = HeapNumber::kValueOffset - kHeapObjectTag);
499 417
500 void TruncateHeapNumberToI(Register result_reg, Register input_reg); 418 void TruncateHeapNumberToI(Register result_reg, Register input_reg);
501 void TruncateDoubleToI(Register result_reg, XMMRegister input_reg); 419 void TruncateDoubleToI(Register result_reg, XMMRegister input_reg);
502 420
503 void DoubleToI(Register result_reg, XMMRegister input_reg, 421 void DoubleToI(Register result_reg, XMMRegister input_reg,
504 XMMRegister scratch, MinusZeroMode minus_zero_mode, 422 XMMRegister scratch, MinusZeroMode minus_zero_mode,
(...skipping 17 matching lines...) Expand all
522 STATIC_ASSERT(kSmiTag == 0); 440 STATIC_ASSERT(kSmiTag == 0);
523 j(not_carry, is_smi); 441 j(not_carry, is_smi);
524 } 442 }
525 443
526 void LoadUint32(XMMRegister dst, Register src) { 444 void LoadUint32(XMMRegister dst, Register src) {
527 LoadUint32(dst, Operand(src)); 445 LoadUint32(dst, Operand(src));
528 } 446 }
529 void LoadUint32(XMMRegister dst, const Operand& src); 447 void LoadUint32(XMMRegister dst, const Operand& src);
530 448
531 // Jump the register contains a smi. 449 // Jump the register contains a smi.
532 inline void JumpIfSmi(Register value, 450 inline void JumpIfSmi(Register value, Label* smi_label,
533 Label* smi_label,
534 Label::Distance distance = Label::kFar) { 451 Label::Distance distance = Label::kFar) {
535 test(value, Immediate(kSmiTagMask)); 452 test(value, Immediate(kSmiTagMask));
536 j(zero, smi_label, distance); 453 j(zero, smi_label, distance);
537 } 454 }
538 // Jump if the operand is a smi. 455 // Jump if the operand is a smi.
539 inline void JumpIfSmi(Operand value, 456 inline void JumpIfSmi(Operand value, Label* smi_label,
540 Label* smi_label,
541 Label::Distance distance = Label::kFar) { 457 Label::Distance distance = Label::kFar) {
542 test(value, Immediate(kSmiTagMask)); 458 test(value, Immediate(kSmiTagMask));
543 j(zero, smi_label, distance); 459 j(zero, smi_label, distance);
544 } 460 }
545 // Jump if register contain a non-smi. 461 // Jump if register contain a non-smi.
546 inline void JumpIfNotSmi(Register value, 462 inline void JumpIfNotSmi(Register value, Label* not_smi_label,
547 Label* not_smi_label,
548 Label::Distance distance = Label::kFar) { 463 Label::Distance distance = Label::kFar) {
549 test(value, Immediate(kSmiTagMask)); 464 test(value, Immediate(kSmiTagMask));
550 j(not_zero, not_smi_label, distance); 465 j(not_zero, not_smi_label, distance);
551 } 466 }
552 467
553 void LoadInstanceDescriptors(Register map, Register descriptors); 468 void LoadInstanceDescriptors(Register map, Register descriptors);
554 void EnumLength(Register dst, Register map); 469 void EnumLength(Register dst, Register map);
555 void NumberOfOwnDescriptors(Register dst, Register map); 470 void NumberOfOwnDescriptors(Register dst, Register map);
556 void LoadAccessor(Register dst, Register holder, int accessor_index, 471 void LoadAccessor(Register dst, Register holder, int accessor_index,
557 AccessorComponent accessor); 472 AccessorComponent accessor);
(...skipping 54 matching lines...) Expand 10 before | Expand all | Expand 10 after
612 527
613 // Unlink the stack handler on top of the stack from the stack handler chain. 528 // Unlink the stack handler on top of the stack from the stack handler chain.
614 void PopStackHandler(); 529 void PopStackHandler();
615 530
616 // --------------------------------------------------------------------------- 531 // ---------------------------------------------------------------------------
617 // Inline caching support 532 // Inline caching support
618 533
619 // Generate code for checking access rights - used for security checks 534 // Generate code for checking access rights - used for security checks
620 // on access to global objects across environments. The holder register 535 // on access to global objects across environments. The holder register
621 // is left untouched, but the scratch register is clobbered. 536 // is left untouched, but the scratch register is clobbered.
622 void CheckAccessGlobalProxy(Register holder_reg, 537 void CheckAccessGlobalProxy(Register holder_reg, Register scratch1,
623 Register scratch1, 538 Register scratch2, Label* miss);
624 Register scratch2,
625 Label* miss);
626 539
627 void GetNumberHash(Register r0, Register scratch); 540 void GetNumberHash(Register r0, Register scratch);
628 541
629 void LoadFromNumberDictionary(Label* miss, 542 void LoadFromNumberDictionary(Label* miss, Register elements, Register key,
630 Register elements, 543 Register r0, Register r1, Register r2,
631 Register key,
632 Register r0,
633 Register r1,
634 Register r2,
635 Register result); 544 Register result);
636 545
637
638 // --------------------------------------------------------------------------- 546 // ---------------------------------------------------------------------------
639 // Allocation support 547 // Allocation support
640 548
641 // Allocate an object in new space or old space. If the given space 549 // Allocate an object in new space or old space. If the given space
642 // is exhausted control continues at the gc_required label. The allocated 550 // is exhausted control continues at the gc_required label. The allocated
643 // object is returned in result and end of the new object is returned in 551 // object is returned in result and end of the new object is returned in
644 // result_end. The register scratch can be passed as no_reg in which case 552 // result_end. The register scratch can be passed as no_reg in which case
645 // an additional object reference will be added to the reloc info. The 553 // an additional object reference will be added to the reloc info. The
646 // returned pointers in result and result_end have not yet been tagged as 554 // returned pointers in result and result_end have not yet been tagged as
647 // heap objects. If result_contains_top_on_entry is true the content of 555 // heap objects. If result_contains_top_on_entry is true the content of
648 // result is known to be the allocation top on entry (could be result_end 556 // result is known to be the allocation top on entry (could be result_end
649 // from a previous call). If result_contains_top_on_entry is true scratch 557 // from a previous call). If result_contains_top_on_entry is true scratch
650 // should be no_reg as it is never used. 558 // should be no_reg as it is never used.
651 void Allocate(int object_size, 559 void Allocate(int object_size, Register result, Register result_end,
652 Register result, 560 Register scratch, Label* gc_required, AllocationFlags flags);
653 Register result_end,
654 Register scratch,
655 Label* gc_required,
656 AllocationFlags flags);
657 561
658 void Allocate(int header_size, 562 void Allocate(int header_size, ScaleFactor element_size,
659 ScaleFactor element_size, 563 Register element_count, RegisterValueType element_count_type,
660 Register element_count, 564 Register result, Register result_end, Register scratch,
661 RegisterValueType element_count_type, 565 Label* gc_required, AllocationFlags flags);
662 Register result,
663 Register result_end,
664 Register scratch,
665 Label* gc_required,
666 AllocationFlags flags);
667 566
668 void Allocate(Register object_size, 567 void Allocate(Register object_size, Register result, Register result_end,
669 Register result, 568 Register scratch, Label* gc_required, AllocationFlags flags);
670 Register result_end,
671 Register scratch,
672 Label* gc_required,
673 AllocationFlags flags);
674 569
675 // Allocate a heap number in new space with undefined value. The 570 // Allocate a heap number in new space with undefined value. The
676 // register scratch2 can be passed as no_reg; the others must be 571 // register scratch2 can be passed as no_reg; the others must be
677 // valid registers. Returns tagged pointer in result register, or 572 // valid registers. Returns tagged pointer in result register, or
678 // jumps to gc_required if new space is full. 573 // jumps to gc_required if new space is full.
679 void AllocateHeapNumber(Register result, 574 void AllocateHeapNumber(Register result, Register scratch1, Register scratch2,
680 Register scratch1, 575 Label* gc_required, MutableMode mode = IMMUTABLE);
681 Register scratch2,
682 Label* gc_required,
683 MutableMode mode = IMMUTABLE);
684 576
685 // Allocate a sequential string. All the header fields of the string object 577 // Allocate a sequential string. All the header fields of the string object
686 // are initialized. 578 // are initialized.
687 void AllocateTwoByteString(Register result, 579 void AllocateTwoByteString(Register result, Register length,
688 Register length, 580 Register scratch1, Register scratch2,
689 Register scratch1, 581 Register scratch3, Label* gc_required);
690 Register scratch2,
691 Register scratch3,
692 Label* gc_required);
693 void AllocateOneByteString(Register result, Register length, 582 void AllocateOneByteString(Register result, Register length,
694 Register scratch1, Register scratch2, 583 Register scratch1, Register scratch2,
695 Register scratch3, Label* gc_required); 584 Register scratch3, Label* gc_required);
696 void AllocateOneByteString(Register result, int length, Register scratch1, 585 void AllocateOneByteString(Register result, int length, Register scratch1,
697 Register scratch2, Label* gc_required); 586 Register scratch2, Label* gc_required);
698 587
699 // Allocate a raw cons string object. Only the map field of the result is 588 // Allocate a raw cons string object. Only the map field of the result is
700 // initialized. 589 // initialized.
701 void AllocateTwoByteConsString(Register result, 590 void AllocateTwoByteConsString(Register result, Register scratch1,
702 Register scratch1, 591 Register scratch2, Label* gc_required);
703 Register scratch2,
704 Label* gc_required);
705 void AllocateOneByteConsString(Register result, Register scratch1, 592 void AllocateOneByteConsString(Register result, Register scratch1,
706 Register scratch2, Label* gc_required); 593 Register scratch2, Label* gc_required);
707 594
708 // Allocate a raw sliced string object. Only the map field of the result is 595 // Allocate a raw sliced string object. Only the map field of the result is
709 // initialized. 596 // initialized.
710 void AllocateTwoByteSlicedString(Register result, 597 void AllocateTwoByteSlicedString(Register result, Register scratch1,
711 Register scratch1, 598 Register scratch2, Label* gc_required);
712 Register scratch2,
713 Label* gc_required);
714 void AllocateOneByteSlicedString(Register result, Register scratch1, 599 void AllocateOneByteSlicedString(Register result, Register scratch1,
715 Register scratch2, Label* gc_required); 600 Register scratch2, Label* gc_required);
716 601
717 // Copy memory, byte-by-byte, from source to destination. Not optimized for 602 // Copy memory, byte-by-byte, from source to destination. Not optimized for
718 // long or aligned copies. 603 // long or aligned copies.
719 // The contents of index and scratch are destroyed. 604 // The contents of index and scratch are destroyed.
720 void CopyBytes(Register source, 605 void CopyBytes(Register source, Register destination, Register length,
721 Register destination,
722 Register length,
723 Register scratch); 606 Register scratch);
724 607
725 // Initialize fields with filler values. Fields starting at |current_address| 608 // Initialize fields with filler values. Fields starting at |current_address|
726 // not including |end_address| are overwritten with the value in |filler|. At 609 // not including |end_address| are overwritten with the value in |filler|. At
727 // the end the loop, |current_address| takes the value of |end_address|. 610 // the end the loop, |current_address| takes the value of |end_address|.
728 void InitializeFieldsWithFiller(Register current_address, 611 void InitializeFieldsWithFiller(Register current_address,
729 Register end_address, Register filler); 612 Register end_address, Register filler);
730 613
731 // --------------------------------------------------------------------------- 614 // ---------------------------------------------------------------------------
732 // Support functions. 615 // Support functions.
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after
766 // Call a code stub. Generate the code if necessary. 649 // Call a code stub. Generate the code if necessary.
767 void CallStub(CodeStub* stub, TypeFeedbackId ast_id = TypeFeedbackId::None()); 650 void CallStub(CodeStub* stub, TypeFeedbackId ast_id = TypeFeedbackId::None());
768 651
769 // Tail call a code stub (jump). Generate the code if necessary. 652 // Tail call a code stub (jump). Generate the code if necessary.
770 void TailCallStub(CodeStub* stub); 653 void TailCallStub(CodeStub* stub);
771 654
772 // Return from a code stub after popping its arguments. 655 // Return from a code stub after popping its arguments.
773 void StubReturn(int argc); 656 void StubReturn(int argc);
774 657
775 // Call a runtime routine. 658 // Call a runtime routine.
776 void CallRuntime(const Runtime::Function* f, 659 void CallRuntime(const Runtime::Function* f, int num_arguments,
777 int num_arguments,
778 SaveFPRegsMode save_doubles = kDontSaveFPRegs); 660 SaveFPRegsMode save_doubles = kDontSaveFPRegs);
779 void CallRuntimeSaveDoubles(Runtime::FunctionId id) { 661 void CallRuntimeSaveDoubles(Runtime::FunctionId id) {
780 const Runtime::Function* function = Runtime::FunctionForId(id); 662 const Runtime::Function* function = Runtime::FunctionForId(id);
781 CallRuntime(function, function->nargs, kSaveFPRegs); 663 CallRuntime(function, function->nargs, kSaveFPRegs);
782 } 664 }
783 665
784 // Convenience function: Same as above, but takes the fid instead. 666 // Convenience function: Same as above, but takes the fid instead.
785 void CallRuntime(Runtime::FunctionId id, 667 void CallRuntime(Runtime::FunctionId id, int num_arguments,
786 int num_arguments,
787 SaveFPRegsMode save_doubles = kDontSaveFPRegs) { 668 SaveFPRegsMode save_doubles = kDontSaveFPRegs) {
788 CallRuntime(Runtime::FunctionForId(id), num_arguments, save_doubles); 669 CallRuntime(Runtime::FunctionForId(id), num_arguments, save_doubles);
789 } 670 }
790 671
791 // Convenience function: call an external reference. 672 // Convenience function: call an external reference.
792 void CallExternalReference(ExternalReference ref, int num_arguments); 673 void CallExternalReference(ExternalReference ref, int num_arguments);
793 674
794 // Tail call of a runtime routine (jump). 675 // Tail call of a runtime routine (jump).
795 // Like JumpToExternalReference, but also takes care of passing the number 676 // Like JumpToExternalReference, but also takes care of passing the number
796 // of parameters. 677 // of parameters.
797 void TailCallExternalReference(const ExternalReference& ext, 678 void TailCallExternalReference(const ExternalReference& ext,
798 int num_arguments, 679 int num_arguments, int result_size);
799 int result_size);
800 680
801 // Convenience function: tail call a runtime routine (jump). 681 // Convenience function: tail call a runtime routine (jump).
802 void TailCallRuntime(Runtime::FunctionId fid, 682 void TailCallRuntime(Runtime::FunctionId fid, int num_arguments,
803 int num_arguments,
804 int result_size); 683 int result_size);
805 684
806 // Before calling a C-function from generated code, align arguments on stack. 685 // Before calling a C-function from generated code, align arguments on stack.
807 // After aligning the frame, arguments must be stored in esp[0], esp[4], 686 // After aligning the frame, arguments must be stored in esp[0], esp[4],
808 // etc., not pushed. The argument count assumes all arguments are word sized. 687 // etc., not pushed. The argument count assumes all arguments are word sized.
809 // Some compilers/platforms require the stack to be aligned when calling 688 // Some compilers/platforms require the stack to be aligned when calling
810 // C++ code. 689 // C++ code.
811 // Needs a scratch register to do some arithmetic. This register will be 690 // Needs a scratch register to do some arithmetic. This register will be
812 // trashed. 691 // trashed.
813 void PrepareCallCFunction(int num_arguments, Register scratch); 692 void PrepareCallCFunction(int num_arguments, Register scratch);
(...skipping 82 matching lines...) Expand 10 before | Expand all | Expand 10 after
896 775
897 // --------------------------------------------------------------------------- 776 // ---------------------------------------------------------------------------
898 // StatsCounter support 777 // StatsCounter support
899 778
900 void SetCounter(StatsCounter* counter, int value); 779 void SetCounter(StatsCounter* counter, int value);
901 void IncrementCounter(StatsCounter* counter, int value); 780 void IncrementCounter(StatsCounter* counter, int value);
902 void DecrementCounter(StatsCounter* counter, int value); 781 void DecrementCounter(StatsCounter* counter, int value);
903 void IncrementCounter(Condition cc, StatsCounter* counter, int value); 782 void IncrementCounter(Condition cc, StatsCounter* counter, int value);
904 void DecrementCounter(Condition cc, StatsCounter* counter, int value); 783 void DecrementCounter(Condition cc, StatsCounter* counter, int value);
905 784
906
907 // --------------------------------------------------------------------------- 785 // ---------------------------------------------------------------------------
908 // Debugging 786 // Debugging
909 787
910 // Calls Abort(msg) if the condition cc is not satisfied. 788 // Calls Abort(msg) if the condition cc is not satisfied.
911 // Use --debug_code to enable. 789 // Use --debug_code to enable.
912 void Assert(Condition cc, BailoutReason reason); 790 void Assert(Condition cc, BailoutReason reason);
913 791
914 void AssertFastElements(Register elements); 792 void AssertFastElements(Register elements);
915 793
916 // Like Assert(), but always enabled. 794 // Like Assert(), but always enabled.
(...skipping 30 matching lines...) Expand all
947 825
948 // Checks if the given register or operand is a unique name 826 // Checks if the given register or operand is a unique name
949 void JumpIfNotUniqueNameInstanceType(Register reg, Label* not_unique_name, 827 void JumpIfNotUniqueNameInstanceType(Register reg, Label* not_unique_name,
950 Label::Distance distance = Label::kFar) { 828 Label::Distance distance = Label::kFar) {
951 JumpIfNotUniqueNameInstanceType(Operand(reg), not_unique_name, distance); 829 JumpIfNotUniqueNameInstanceType(Operand(reg), not_unique_name, distance);
952 } 830 }
953 831
954 void JumpIfNotUniqueNameInstanceType(Operand operand, Label* not_unique_name, 832 void JumpIfNotUniqueNameInstanceType(Operand operand, Label* not_unique_name,
955 Label::Distance distance = Label::kFar); 833 Label::Distance distance = Label::kFar);
956 834
957 void EmitSeqStringSetCharCheck(Register string, 835 void EmitSeqStringSetCharCheck(Register string, Register index,
958 Register index, 836 Register value, uint32_t encoding_mask);
959 Register value,
960 uint32_t encoding_mask);
961 837
962 static int SafepointRegisterStackIndex(Register reg) { 838 static int SafepointRegisterStackIndex(Register reg) {
963 return SafepointRegisterStackIndex(reg.code()); 839 return SafepointRegisterStackIndex(reg.code());
964 } 840 }
965 841
966 // Load the type feedback vector from a JavaScript frame. 842 // Load the type feedback vector from a JavaScript frame.
967 void EmitLoadTypeFeedbackVector(Register vector); 843 void EmitLoadTypeFeedbackVector(Register vector);
968 844
969 // Activation support. 845 // Activation support.
970 void EnterFrame(StackFrame::Type type); 846 void EnterFrame(StackFrame::Type type);
(...skipping 29 matching lines...) Expand all
1000 Register scratch1, Label* found); 876 Register scratch1, Label* found);
1001 877
1002 private: 878 private:
1003 bool generating_stub_; 879 bool generating_stub_;
1004 bool has_frame_; 880 bool has_frame_;
1005 // This handle will be patched with the code object on installation. 881 // This handle will be patched with the code object on installation.
1006 Handle<Object> code_object_; 882 Handle<Object> code_object_;
1007 883
1008 // Helper functions for generating invokes. 884 // Helper functions for generating invokes.
1009 void InvokePrologue(const ParameterCount& expected, 885 void InvokePrologue(const ParameterCount& expected,
1010 const ParameterCount& actual, 886 const ParameterCount& actual, Label* done,
1011 Label* done, 887 bool* definitely_mismatches, InvokeFlag flag,
1012 bool* definitely_mismatches,
1013 InvokeFlag flag,
1014 Label::Distance done_distance, 888 Label::Distance done_distance,
1015 const CallWrapper& call_wrapper); 889 const CallWrapper& call_wrapper);
1016 890
891 void FloodFunctionIfStepping(Register fun, Register new_target,
892 const ParameterCount& expected,
893 const ParameterCount& actual);
894
1017 void EnterExitFramePrologue(); 895 void EnterExitFramePrologue();
1018 void EnterExitFrameEpilogue(int argc, bool save_doubles); 896 void EnterExitFrameEpilogue(int argc, bool save_doubles);
1019 897
1020 void LeaveExitFrameEpilogue(bool restore_context); 898 void LeaveExitFrameEpilogue(bool restore_context);
1021 899
1022 // Allocation support helpers. 900 // Allocation support helpers.
1023 void LoadAllocationTopHelper(Register result, 901 void LoadAllocationTopHelper(Register result, Register scratch,
1024 Register scratch,
1025 AllocationFlags flags); 902 AllocationFlags flags);
1026 903
1027 void UpdateAllocationTopHelper(Register result_end, 904 void UpdateAllocationTopHelper(Register result_end, Register scratch,
1028 Register scratch,
1029 AllocationFlags flags); 905 AllocationFlags flags);
1030 906
1031 // Helper for implementing JumpIfNotInNewSpace and JumpIfInNewSpace. 907 // Helper for implementing JumpIfNotInNewSpace and JumpIfInNewSpace.
1032 void InNewSpace(Register object, 908 void InNewSpace(Register object, Register scratch, Condition cc,
1033 Register scratch,
1034 Condition cc,
1035 Label* condition_met, 909 Label* condition_met,
1036 Label::Distance condition_met_distance = Label::kFar); 910 Label::Distance condition_met_distance = Label::kFar);
1037 911
1038 // Helper for finding the mark bits for an address. Afterwards, the 912 // Helper for finding the mark bits for an address. Afterwards, the
1039 // bitmap register points at the word with the mark bits and the mask 913 // bitmap register points at the word with the mark bits and the mask
1040 // the position of the first bit. Uses ecx as scratch and leaves addr_reg 914 // the position of the first bit. Uses ecx as scratch and leaves addr_reg
1041 // unchanged. 915 // unchanged.
1042 inline void GetMarkBits(Register addr_reg, 916 inline void GetMarkBits(Register addr_reg, Register bitmap_reg,
1043 Register bitmap_reg,
1044 Register mask_reg); 917 Register mask_reg);
1045 918
1046 // Compute memory operands for safepoint stack slots. 919 // Compute memory operands for safepoint stack slots.
1047 Operand SafepointRegisterSlot(Register reg); 920 Operand SafepointRegisterSlot(Register reg);
1048 static int SafepointRegisterStackIndex(int reg_code); 921 static int SafepointRegisterStackIndex(int reg_code);
1049 922
1050 // Needs access to SafepointRegisterStackIndex for compiled frame 923 // Needs access to SafepointRegisterStackIndex for compiled frame
1051 // traversal. 924 // traversal.
1052 friend class StandardFrame; 925 friend class StandardFrame;
1053 }; 926 };
1054 927
1055
1056 // The code patcher is used to patch (typically) small parts of code e.g. for 928 // The code patcher is used to patch (typically) small parts of code e.g. for
1057 // debugging and other types of instrumentation. When using the code patcher 929 // debugging and other types of instrumentation. When using the code patcher
1058 // the exact number of bytes specified must be emitted. Is not legal to emit 930 // the exact number of bytes specified must be emitted. Is not legal to emit
1059 // relocation information. If any of these constraints are violated it causes 931 // relocation information. If any of these constraints are violated it causes
1060 // an assertion. 932 // an assertion.
1061 class CodePatcher { 933 class CodePatcher {
1062 public: 934 public:
1063 CodePatcher(byte* address, int size); 935 CodePatcher(byte* address, int size);
1064 ~CodePatcher(); 936 ~CodePatcher();
1065 937
1066 // Macro assembler to emit code. 938 // Macro assembler to emit code.
1067 MacroAssembler* masm() { return &masm_; } 939 MacroAssembler* masm() { return &masm_; }
1068 940
1069 private: 941 private:
1070 byte* address_; // The address of the code being patched. 942 byte* address_; // The address of the code being patched.
1071 int size_; // Number of bytes of the expected patch size. 943 int size_; // Number of bytes of the expected patch size.
1072 MacroAssembler masm_; // Macro assembler used to generate the code. 944 MacroAssembler masm_; // Macro assembler used to generate the code.
1073 }; 945 };
1074 946
1075
1076 // ----------------------------------------------------------------------------- 947 // -----------------------------------------------------------------------------
1077 // Static helper functions. 948 // Static helper functions.
1078 949
1079 // Generate an Operand for loading a field from an object. 950 // Generate an Operand for loading a field from an object.
1080 inline Operand FieldOperand(Register object, int offset) { 951 inline Operand FieldOperand(Register object, int offset) {
1081 return Operand(object, offset - kHeapObjectTag); 952 return Operand(object, offset - kHeapObjectTag);
1082 } 953 }
1083 954
1084
1085 // Generate an Operand for loading an indexed field from an object. 955 // Generate an Operand for loading an indexed field from an object.
1086 inline Operand FieldOperand(Register object, 956 inline Operand FieldOperand(Register object, Register index, ScaleFactor scale,
1087 Register index,
1088 ScaleFactor scale,
1089 int offset) { 957 int offset) {
1090 return Operand(object, index, scale, offset - kHeapObjectTag); 958 return Operand(object, index, scale, offset - kHeapObjectTag);
1091 } 959 }
1092 960
1093 961 inline Operand FixedArrayElementOperand(Register array, Register index_as_smi,
1094 inline Operand FixedArrayElementOperand(Register array,
1095 Register index_as_smi,
1096 int additional_offset = 0) { 962 int additional_offset = 0) {
1097 int offset = FixedArray::kHeaderSize + additional_offset * kPointerSize; 963 int offset = FixedArray::kHeaderSize + additional_offset * kPointerSize;
1098 return FieldOperand(array, index_as_smi, times_half_pointer_size, offset); 964 return FieldOperand(array, index_as_smi, times_half_pointer_size, offset);
1099 } 965 }
1100 966
1101
1102 inline Operand ContextOperand(Register context, int index) { 967 inline Operand ContextOperand(Register context, int index) {
1103 return Operand(context, Context::SlotOffset(index)); 968 return Operand(context, Context::SlotOffset(index));
1104 } 969 }
1105 970
1106
1107 inline Operand ContextOperand(Register context, Register index) { 971 inline Operand ContextOperand(Register context, Register index) {
1108 return Operand(context, index, times_pointer_size, Context::SlotOffset(0)); 972 return Operand(context, index, times_pointer_size, Context::SlotOffset(0));
1109 } 973 }
1110 974
1111
1112 inline Operand GlobalObjectOperand() { 975 inline Operand GlobalObjectOperand() {
1113 return ContextOperand(esi, Context::GLOBAL_OBJECT_INDEX); 976 return ContextOperand(esi, Context::GLOBAL_OBJECT_INDEX);
1114 } 977 }
1115 978
1116
1117 #ifdef GENERATED_CODE_COVERAGE 979 #ifdef GENERATED_CODE_COVERAGE
1118 extern void LogGeneratedCodeCoverage(const char* file_line); 980 extern void LogGeneratedCodeCoverage(const char* file_line);
1119 #define CODE_COVERAGE_STRINGIFY(x) #x 981 #define CODE_COVERAGE_STRINGIFY(x) #x
1120 #define CODE_COVERAGE_TOSTRING(x) CODE_COVERAGE_STRINGIFY(x) 982 #define CODE_COVERAGE_TOSTRING(x) CODE_COVERAGE_STRINGIFY(x)
1121 #define __FILE_LINE__ __FILE__ ":" CODE_COVERAGE_TOSTRING(__LINE__) 983 #define __FILE_LINE__ __FILE__ ":" CODE_COVERAGE_TOSTRING(__LINE__)
1122 #define ACCESS_MASM(masm) { \ 984 #define ACCESS_MASM(masm) { \
1123 byte* ia32_coverage_function = \ 985 byte* ia32_coverage_function = \
1124 reinterpret_cast<byte*>(FUNCTION_ADDR(LogGeneratedCodeCoverage)); \ 986 reinterpret_cast<byte*>(FUNCTION_ADDR(LogGeneratedCodeCoverage)); \
1125 masm->pushfd(); \ 987 masm->pushfd(); \
1126 masm->pushad(); \ 988 masm->pushad(); \
1127 masm->push(Immediate(reinterpret_cast<int>(&__FILE_LINE__))); \ 989 masm->push(Immediate(reinterpret_cast<int>(&__FILE_LINE__))); \
1128 masm->call(ia32_coverage_function, RelocInfo::RUNTIME_ENTRY); \ 990 masm->call(ia32_coverage_function, RelocInfo::RUNTIME_ENTRY); \
1129 masm->pop(eax); \ 991 masm->pop(eax); \
1130 masm->popad(); \ 992 masm->popad(); \
1131 masm->popfd(); \ 993 masm->popfd(); \
1132 } \ 994 } \
1133 masm-> 995 masm->
1134 #else 996 #else
1135 #define ACCESS_MASM(masm) masm-> 997 #define ACCESS_MASM(masm) masm->
1136 #endif 998 #endif
1137 999
1138
1139 } // namespace internal 1000 } // namespace internal
1140 } // namespace v8 1001 } // namespace v8
1141 1002
1142 #endif // V8_IA32_MACRO_ASSEMBLER_IA32_H_ 1003 #endif // V8_IA32_MACRO_ASSEMBLER_IA32_H_
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698