Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 1 | // Copyright 2011 Google Inc. All Rights Reserved. |
| 2 | |
| 3 | #include "context_arm.h" |
| 4 | |
| 5 | #include "object.h" |
| 6 | |
| 7 | namespace art { |
| 8 | namespace arm { |
| 9 | |
| 10 | ArmContext::ArmContext() { |
Ian Rogers | 67375ac | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 11 | #ifndef NDEBUG |
Ian Rogers | ad42e13 | 2011-09-17 20:23:33 -0700 | [diff] [blame] | 12 | // Initialize registers with easy to spot debug values |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 13 | for (int i=0; i < 16; i++) { |
Ian Rogers | 67375ac | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 14 | gprs_[i] = 0xEBAD6070+i; |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 15 | } |
Ian Rogers | 15fdb8c | 2011-09-25 15:45:07 -0700 | [diff] [blame^] | 16 | for (int i=0; i < 32; i++) { |
| 17 | fprs_[i] = 0xEBAD8070+i; |
| 18 | } |
Ian Rogers | 67375ac | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 19 | #endif |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 20 | } |
| 21 | |
| 22 | void ArmContext::FillCalleeSaves(const Frame& fr) { |
| 23 | Method* method = fr.GetMethod(); |
| 24 | uint32_t core_spills = method->GetCoreSpillMask(); |
Ian Rogers | 15fdb8c | 2011-09-25 15:45:07 -0700 | [diff] [blame^] | 25 | uint32_t fp_core_spills = method->GetFpSpillMask(); |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 26 | size_t spill_count = __builtin_popcount(core_spills); |
Ian Rogers | 15fdb8c | 2011-09-25 15:45:07 -0700 | [diff] [blame^] | 27 | size_t fp_spill_count = __builtin_popcount(fp_core_spills); |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 28 | if (spill_count > 0) { |
| 29 | // Lowest number spill is furthest away, walk registers and fill into context |
| 30 | int j = 1; |
| 31 | for(int i = 0; i < 16; i++) { |
| 32 | if (((core_spills >> i) & 1) != 0) { |
| 33 | gprs_[i] = fr.LoadCalleeSave(spill_count - j); |
| 34 | j++; |
| 35 | } |
| 36 | } |
| 37 | } |
Ian Rogers | 15fdb8c | 2011-09-25 15:45:07 -0700 | [diff] [blame^] | 38 | if (fp_spill_count > 0) { |
| 39 | // Lowest number spill is furthest away, walk registers and fill into context |
| 40 | int j = 1; |
| 41 | for(int i = 0; i < 32; i++) { |
| 42 | if (((fp_core_spills >> i) & 1) != 0) { |
| 43 | fprs_[i] = fr.LoadCalleeSave(spill_count + fp_spill_count - j); |
| 44 | j++; |
| 45 | } |
| 46 | } |
| 47 | } |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 48 | } |
| 49 | |
| 50 | void ArmContext::DoLongJump() { |
Elliott Hughes | 85d1545 | 2011-09-16 17:33:01 -0700 | [diff] [blame] | 51 | #if defined(__arm__) |
Ian Rogers | 15fdb8c | 2011-09-25 15:45:07 -0700 | [diff] [blame^] | 52 | // TODO: Load all GPRs, currently R0 to R3 aren't restored |
| 53 | asm volatile ( "vldm %2, {%%s0-%%s31}\n" |
| 54 | "mov %%r0, %0\n" |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 55 | "mov %%r1, %1\n" |
Ian Rogers | 15fdb8c | 2011-09-25 15:45:07 -0700 | [diff] [blame^] | 56 | "ldm %%r0, {%%r4-%%r14}\n" |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 57 | "mov %%pc,%%r1\n" |
| 58 | : // output |
Ian Rogers | 15fdb8c | 2011-09-25 15:45:07 -0700 | [diff] [blame^] | 59 | : "r"(&gprs_[4]), "r"(gprs_[R15]), "r"(&fprs_[S0]) // input |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 60 | :); // clobber |
Elliott Hughes | 85d1545 | 2011-09-16 17:33:01 -0700 | [diff] [blame] | 61 | #else |
| 62 | UNIMPLEMENTED(FATAL); |
| 63 | #endif |
Ian Rogers | bdb0391 | 2011-09-14 00:55:44 -0700 | [diff] [blame] | 64 | } |
| 65 | |
| 66 | } // namespace arm |
| 67 | } // namespace art |