blob: fafc0462d637abb597367e80cf43e694d76ba4c8 [file] [log] [blame]
/*
* Copyright (C) 2016 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.
*/
#ifndef ART_COMPILER_DEBUG_DWARF_EXPRESSION_H_
#define ART_COMPILER_DEBUG_DWARF_EXPRESSION_H_
#include <cstddef>
#include <cstdint>
#include "debug/dwarf/dwarf_constants.h"
#include "debug/dwarf/writer.h"
namespace art {
namespace dwarf {
// Writer for DWARF expressions which are used in .debug_info and .debug_loc sections.
// See the DWARF specification for the precise meaning of the opcodes.
// If multiple equivalent encodings are possible, it will choose the most compact one.
// The writer is not exhaustive - it only implements opcodes we have needed so far.
class Expression : private Writer<> {
public:
using Writer<>::data;
using Writer<>::size;
// Push signed integer on the stack.
void WriteOpConsts(int32_t value) {
if (0 <= value && value < 32) {
PushUint8(DW_OP_lit0 + value);
} else {
PushUint8(DW_OP_consts);
PushSleb128(value);
}
}
// Push unsigned integer on the stack.
void WriteOpConstu(uint32_t value) {
if (value < 32) {
PushUint8(DW_OP_lit0 + value);
} else {
PushUint8(DW_OP_constu);
PushUleb128(value);
}
}
// Variable is stored in given register.
void WriteOpReg(uint32_t dwarf_reg_num) {
if (dwarf_reg_num < 32) {
PushUint8(DW_OP_reg0 + dwarf_reg_num);
} else {
PushUint8(DW_OP_regx);
PushUleb128(dwarf_reg_num);
}
}
// Variable is stored on stack. Also see DW_AT_frame_base.
void WriteOpFbreg(int32_t stack_offset) {
PushUint8(DW_OP_fbreg);
PushSleb128(stack_offset);
}
// The variable is stored in multiple locations (pieces).
void WriteOpPiece(uint32_t num_bytes) {
PushUint8(DW_OP_piece);
PushUleb128(num_bytes);
}
// Loads 32-bit or 64-bit value depending on architecture.
void WriteOpDeref() { PushUint8(DW_OP_deref); }
// Loads value of given byte size.
void WriteOpDerefSize(uint8_t num_bytes) {
PushUint8(DW_OP_deref_size);
PushUint8(num_bytes);
}
// Pop two values and push their sum.
void WriteOpPlus() { PushUint8(DW_OP_plus); }
// Add constant value to value on top of stack.
void WriteOpPlusUconst(uint32_t offset) {
PushUint8(DW_OP_plus_uconst);
PushUleb128(offset);
}
// Negate top of stack.
void WriteOpNeg() { PushUint8(DW_OP_neg); }
// Pop two values and push their bitwise-AND.
void WriteOpAnd() { PushUint8(DW_OP_and); }
// Push stack base pointer as determined from .debug_frame.
void WriteOpCallFrameCfa() { PushUint8(DW_OP_call_frame_cfa); }
// Push address of the variable we are working with.
void WriteOpPushObjectAddress() { PushUint8(DW_OP_push_object_address); }
// Return the top stack as the value of the variable.
// Otherwise, the top of stack is the variable's location.
void WriteOpStackValue() { PushUint8(DW_OP_stack_value); }
explicit Expression(std::vector<uint8_t>* buffer) : Writer<>(buffer) {
buffer->clear();
}
};
} // namespace dwarf
} // namespace art
#endif // ART_COMPILER_DEBUG_DWARF_EXPRESSION_H_