diff options
author | Ian Rogers <irogers@google.com> | 2013-07-26 12:05:57 -0700 |
---|---|---|
committer | Ian Rogers <irogers@google.com> | 2013-07-29 09:52:11 -0700 |
commit | 166db04e259ca51838c311891598664deeed85ad (patch) | |
tree | 5ffa63c1b3bf58e92be2890c54f0ba6fdec2059e /compiler/utils/assembler.cc | |
parent | d41119e863a9f9c5809b7732d30870ffd237b60c (diff) | |
download | art-166db04e259ca51838c311891598664deeed85ad.tar.gz art-166db04e259ca51838c311891598664deeed85ad.tar.bz2 art-166db04e259ca51838c311891598664deeed85ad.zip |
Move assembler out of runtime into compiler/utils.
Other directory layout bits of clean up. There is still work to separate quick
and portable in some files (e.g. argument visitor, proxy..).
Change-Id: If8fecffda8ba5c4c47a035f0c622c538c6b58351
Diffstat (limited to 'compiler/utils/assembler.cc')
-rw-r--r-- | compiler/utils/assembler.cc | 119 |
1 files changed, 119 insertions, 0 deletions
diff --git a/compiler/utils/assembler.cc b/compiler/utils/assembler.cc new file mode 100644 index 000000000..92ce0b800 --- /dev/null +++ b/compiler/utils/assembler.cc @@ -0,0 +1,119 @@ +/* + * Copyright (C) 2011 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. + */ + +#include "assembler.h" + +#include <algorithm> +#include <vector> + +#include "arm/assembler_arm.h" +#include "mips/assembler_mips.h" +#include "x86/assembler_x86.h" +#include "globals.h" +#include "memory_region.h" + +namespace art { + +static byte* NewContents(size_t capacity) { + return new byte[capacity]; +} + + +AssemblerBuffer::AssemblerBuffer() { + static const size_t kInitialBufferCapacity = 4 * KB; + contents_ = NewContents(kInitialBufferCapacity); + cursor_ = contents_; + limit_ = ComputeLimit(contents_, kInitialBufferCapacity); + fixup_ = NULL; + slow_path_ = NULL; +#ifndef NDEBUG + has_ensured_capacity_ = false; + fixups_processed_ = false; +#endif + + // Verify internal state. + CHECK_EQ(Capacity(), kInitialBufferCapacity); + CHECK_EQ(Size(), 0U); +} + + +AssemblerBuffer::~AssemblerBuffer() { + delete[] contents_; +} + + +void AssemblerBuffer::ProcessFixups(const MemoryRegion& region) { + AssemblerFixup* fixup = fixup_; + while (fixup != NULL) { + fixup->Process(region, fixup->position()); + fixup = fixup->previous(); + } +} + + +void AssemblerBuffer::FinalizeInstructions(const MemoryRegion& instructions) { + // Copy the instructions from the buffer. + MemoryRegion from(reinterpret_cast<void*>(contents()), Size()); + instructions.CopyFrom(0, from); + // Process fixups in the instructions. + ProcessFixups(instructions); +#ifndef NDEBUG + fixups_processed_ = true; +#endif +} + + +void AssemblerBuffer::ExtendCapacity() { + size_t old_size = Size(); + size_t old_capacity = Capacity(); + size_t new_capacity = std::min(old_capacity * 2, old_capacity + 1 * MB); + + // Allocate the new data area and copy contents of the old one to it. + byte* new_contents = NewContents(new_capacity); + memmove(reinterpret_cast<void*>(new_contents), + reinterpret_cast<void*>(contents_), + old_size); + + // Compute the relocation delta and switch to the new contents area. + ptrdiff_t delta = new_contents - contents_; + contents_ = new_contents; + + // Update the cursor and recompute the limit. + cursor_ += delta; + limit_ = ComputeLimit(new_contents, new_capacity); + + // Verify internal state. + CHECK_EQ(Capacity(), new_capacity); + CHECK_EQ(Size(), old_size); +} + + +Assembler* Assembler::Create(InstructionSet instruction_set) { + switch (instruction_set) { + case kArm: + case kThumb2: + return new arm::ArmAssembler(); + case kMips: + return new mips::MipsAssembler(); + case kX86: + return new x86::X86Assembler(); + default: + LOG(FATAL) << "Unknown InstructionSet: " << instruction_set; + return NULL; + } +} + +} // namespace art |