2014-05-24 20:50:23 +08:00
|
|
|
//===-- AArch64AsmBackend.cpp - AArch64 Assembler Backend -----------------===//
|
2014-03-29 18:18:08 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
#include "AArch64.h"
|
|
|
|
#include "AArch64RegisterInfo.h"
|
|
|
|
#include "MCTargetDesc/AArch64FixupKinds.h"
|
Re-commit r247683: Replace Triple with a new TargetTuple in MCTargetDesc/* and related. NFC.
Summary:
This is the first patch in the series to migrate Triple's (which are ambiguous)
to TargetTuple's (which aren't).
For the moment, TargetTuple simply passes all requests to the Triple object it
holds. Once it has replaced Triple, it will start to implement the interface in
a more suitable way.
This change makes some changes to the public C++ API. In particular,
InitMCSubtargetInfo(), createMCRelocationInfo(), and createMCSymbolizer()
now take TargetTuples instead of Triples. The other public C++ API's have
been left as-is for the moment to reduce patch size.
This commit also contains a trivial patch to clang to account for the C++ API
change. Thanks go to Pavel Labath for fixing LLDB for me.
Reviewers: rengolin
Subscribers: jyknight, dschuff, arsenm, rampitec, danalbert, srhines, javed.absar, dsanders, echristo, emaste, jholewinski, tberghammer, ted, jfb, llvm-commits, rengolin
Differential Revision: http://reviews.llvm.org/D10969
llvm-svn: 247692
2015-09-15 22:08:28 +08:00
|
|
|
#include "llvm/ADT/TargetTuple.h"
|
2014-03-29 18:18:08 +08:00
|
|
|
#include "llvm/MC/MCAsmBackend.h"
|
|
|
|
#include "llvm/MC/MCDirectives.h"
|
2014-08-07 00:05:02 +08:00
|
|
|
#include "llvm/MC/MCELFObjectWriter.h"
|
2015-01-14 19:23:27 +08:00
|
|
|
#include "llvm/MC/MCFixupKindInfo.h"
|
2014-03-29 18:18:08 +08:00
|
|
|
#include "llvm/MC/MCObjectWriter.h"
|
2014-05-15 00:51:58 +08:00
|
|
|
#include "llvm/MC/MCSectionELF.h"
|
2014-07-25 19:42:14 +08:00
|
|
|
#include "llvm/MC/MCSectionMachO.h"
|
2015-03-25 05:47:03 +08:00
|
|
|
#include "llvm/MC/MCValue.h"
|
2014-03-29 18:18:08 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include "llvm/Support/MachO.h"
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
class AArch64AsmBackend : public MCAsmBackend {
|
2014-03-29 18:18:08 +08:00
|
|
|
static const unsigned PCRelFlagVal =
|
|
|
|
MCFixupKindInfo::FKF_IsAlignedDownTo32Bits | MCFixupKindInfo::FKF_IsPCRel;
|
|
|
|
|
|
|
|
public:
|
2014-05-24 20:50:23 +08:00
|
|
|
AArch64AsmBackend(const Target &T) : MCAsmBackend() {}
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2014-04-29 15:58:25 +08:00
|
|
|
unsigned getNumFixupKinds() const override {
|
2014-05-24 20:50:23 +08:00
|
|
|
return AArch64::NumTargetFixupKinds;
|
2014-04-29 15:58:25 +08:00
|
|
|
}
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2014-04-29 15:58:25 +08:00
|
|
|
const MCFixupKindInfo &getFixupKindInfo(MCFixupKind Kind) const override {
|
2014-05-24 20:50:23 +08:00
|
|
|
const static MCFixupKindInfo Infos[AArch64::NumTargetFixupKinds] = {
|
2014-03-29 18:18:08 +08:00
|
|
|
// This table *must* be in the order that the fixup_* kinds are defined in
|
2014-05-24 20:50:23 +08:00
|
|
|
// AArch64FixupKinds.h.
|
2014-03-29 18:18:08 +08:00
|
|
|
//
|
|
|
|
// Name Offset (bits) Size (bits) Flags
|
2014-05-24 20:50:23 +08:00
|
|
|
{ "fixup_aarch64_pcrel_adr_imm21", 0, 32, PCRelFlagVal },
|
|
|
|
{ "fixup_aarch64_pcrel_adrp_imm21", 0, 32, PCRelFlagVal },
|
|
|
|
{ "fixup_aarch64_add_imm12", 10, 12, 0 },
|
|
|
|
{ "fixup_aarch64_ldst_imm12_scale1", 10, 12, 0 },
|
|
|
|
{ "fixup_aarch64_ldst_imm12_scale2", 10, 12, 0 },
|
|
|
|
{ "fixup_aarch64_ldst_imm12_scale4", 10, 12, 0 },
|
|
|
|
{ "fixup_aarch64_ldst_imm12_scale8", 10, 12, 0 },
|
|
|
|
{ "fixup_aarch64_ldst_imm12_scale16", 10, 12, 0 },
|
|
|
|
{ "fixup_aarch64_ldr_pcrel_imm19", 5, 19, PCRelFlagVal },
|
|
|
|
{ "fixup_aarch64_movw", 5, 16, 0 },
|
|
|
|
{ "fixup_aarch64_pcrel_branch14", 5, 14, PCRelFlagVal },
|
|
|
|
{ "fixup_aarch64_pcrel_branch19", 5, 19, PCRelFlagVal },
|
|
|
|
{ "fixup_aarch64_pcrel_branch26", 0, 26, PCRelFlagVal },
|
|
|
|
{ "fixup_aarch64_pcrel_call26", 0, 26, PCRelFlagVal },
|
|
|
|
{ "fixup_aarch64_tlsdesc_call", 0, 0, 0 }
|
2014-03-29 18:18:08 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
if (Kind < FirstTargetFixupKind)
|
|
|
|
return MCAsmBackend::getFixupKindInfo(Kind);
|
|
|
|
|
|
|
|
assert(unsigned(Kind - FirstTargetFixupKind) < getNumFixupKinds() &&
|
|
|
|
"Invalid kind!");
|
|
|
|
return Infos[Kind - FirstTargetFixupKind];
|
|
|
|
}
|
|
|
|
|
|
|
|
void applyFixup(const MCFixup &Fixup, char *Data, unsigned DataSize,
|
2014-04-29 15:58:25 +08:00
|
|
|
uint64_t Value, bool IsPCRel) const override;
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2014-04-29 15:58:25 +08:00
|
|
|
bool mayNeedRelaxation(const MCInst &Inst) const override;
|
2014-03-29 18:18:08 +08:00
|
|
|
bool fixupNeedsRelaxation(const MCFixup &Fixup, uint64_t Value,
|
|
|
|
const MCRelaxableFragment *DF,
|
2014-04-29 15:58:25 +08:00
|
|
|
const MCAsmLayout &Layout) const override;
|
|
|
|
void relaxInstruction(const MCInst &Inst, MCInst &Res) const override;
|
|
|
|
bool writeNopData(uint64_t Count, MCObjectWriter *OW) const override;
|
2014-03-29 18:18:08 +08:00
|
|
|
|
|
|
|
void HandleAssemblerFlag(MCAssemblerFlag Flag) {}
|
|
|
|
|
|
|
|
unsigned getPointerSize() const { return 8; }
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
/// \brief The number of bytes the fixup may change.
|
|
|
|
static unsigned getFixupKindNumBytes(unsigned Kind) {
|
|
|
|
switch (Kind) {
|
|
|
|
default:
|
2014-06-18 13:05:13 +08:00
|
|
|
llvm_unreachable("Unknown fixup kind!");
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_tlsdesc_call:
|
2014-03-29 18:18:08 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
case FK_Data_1:
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
case FK_Data_2:
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_movw:
|
2014-03-29 18:18:08 +08:00
|
|
|
return 2;
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_pcrel_branch14:
|
|
|
|
case AArch64::fixup_aarch64_add_imm12:
|
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale1:
|
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale2:
|
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale4:
|
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale8:
|
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale16:
|
|
|
|
case AArch64::fixup_aarch64_ldr_pcrel_imm19:
|
|
|
|
case AArch64::fixup_aarch64_pcrel_branch19:
|
2014-03-29 18:18:08 +08:00
|
|
|
return 3;
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_pcrel_adr_imm21:
|
|
|
|
case AArch64::fixup_aarch64_pcrel_adrp_imm21:
|
|
|
|
case AArch64::fixup_aarch64_pcrel_branch26:
|
|
|
|
case AArch64::fixup_aarch64_pcrel_call26:
|
2014-03-29 18:18:08 +08:00
|
|
|
case FK_Data_4:
|
|
|
|
return 4;
|
|
|
|
|
|
|
|
case FK_Data_8:
|
|
|
|
return 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned AdrImmBits(unsigned Value) {
|
|
|
|
unsigned lo2 = Value & 0x3;
|
|
|
|
unsigned hi19 = (Value & 0x1ffffc) >> 2;
|
|
|
|
return (hi19 << 5) | (lo2 << 29);
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint64_t adjustFixupValue(unsigned Kind, uint64_t Value) {
|
|
|
|
int64_t SignedValue = static_cast<int64_t>(Value);
|
|
|
|
switch (Kind) {
|
|
|
|
default:
|
2015-01-05 18:15:49 +08:00
|
|
|
llvm_unreachable("Unknown fixup kind!");
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_pcrel_adr_imm21:
|
2014-03-29 18:18:08 +08:00
|
|
|
if (SignedValue > 2097151 || SignedValue < -2097152)
|
|
|
|
report_fatal_error("fixup value out of range");
|
|
|
|
return AdrImmBits(Value & 0x1fffffULL);
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_pcrel_adrp_imm21:
|
2014-03-29 18:18:08 +08:00
|
|
|
return AdrImmBits((Value & 0x1fffff000ULL) >> 12);
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_ldr_pcrel_imm19:
|
|
|
|
case AArch64::fixup_aarch64_pcrel_branch19:
|
2014-03-29 18:18:08 +08:00
|
|
|
// Signed 21-bit immediate
|
|
|
|
if (SignedValue > 2097151 || SignedValue < -2097152)
|
|
|
|
report_fatal_error("fixup value out of range");
|
|
|
|
// Low two bits are not encoded.
|
|
|
|
return (Value >> 2) & 0x7ffff;
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_add_imm12:
|
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale1:
|
2014-03-29 18:18:08 +08:00
|
|
|
// Unsigned 12-bit immediate
|
|
|
|
if (Value >= 0x1000)
|
|
|
|
report_fatal_error("invalid imm12 fixup value");
|
|
|
|
return Value;
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale2:
|
2014-03-29 18:18:08 +08:00
|
|
|
// Unsigned 12-bit immediate which gets multiplied by 2
|
|
|
|
if (Value & 1 || Value >= 0x2000)
|
|
|
|
report_fatal_error("invalid imm12 fixup value");
|
|
|
|
return Value >> 1;
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale4:
|
2014-03-29 18:18:08 +08:00
|
|
|
// Unsigned 12-bit immediate which gets multiplied by 4
|
|
|
|
if (Value & 3 || Value >= 0x4000)
|
|
|
|
report_fatal_error("invalid imm12 fixup value");
|
|
|
|
return Value >> 2;
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale8:
|
2014-03-29 18:18:08 +08:00
|
|
|
// Unsigned 12-bit immediate which gets multiplied by 8
|
|
|
|
if (Value & 7 || Value >= 0x8000)
|
|
|
|
report_fatal_error("invalid imm12 fixup value");
|
|
|
|
return Value >> 3;
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_ldst_imm12_scale16:
|
2014-03-29 18:18:08 +08:00
|
|
|
// Unsigned 12-bit immediate which gets multiplied by 16
|
|
|
|
if (Value & 15 || Value >= 0x10000)
|
|
|
|
report_fatal_error("invalid imm12 fixup value");
|
|
|
|
return Value >> 4;
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_movw:
|
2014-03-29 18:18:08 +08:00
|
|
|
report_fatal_error("no resolvable MOVZ/MOVK fixups supported yet");
|
|
|
|
return Value;
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_pcrel_branch14:
|
2014-03-29 18:18:08 +08:00
|
|
|
// Signed 16-bit immediate
|
|
|
|
if (SignedValue > 32767 || SignedValue < -32768)
|
|
|
|
report_fatal_error("fixup value out of range");
|
|
|
|
// Low two bits are not encoded (4-byte alignment assumed).
|
|
|
|
if (Value & 0x3)
|
|
|
|
report_fatal_error("fixup not sufficiently aligned");
|
|
|
|
return (Value >> 2) & 0x3fff;
|
2014-05-24 20:50:23 +08:00
|
|
|
case AArch64::fixup_aarch64_pcrel_branch26:
|
|
|
|
case AArch64::fixup_aarch64_pcrel_call26:
|
2014-03-29 18:18:08 +08:00
|
|
|
// Signed 28-bit immediate
|
|
|
|
if (SignedValue > 134217727 || SignedValue < -134217728)
|
|
|
|
report_fatal_error("fixup value out of range");
|
|
|
|
// Low two bits are not encoded (4-byte alignment assumed).
|
|
|
|
if (Value & 0x3)
|
|
|
|
report_fatal_error("fixup not sufficiently aligned");
|
|
|
|
return (Value >> 2) & 0x3ffffff;
|
|
|
|
case FK_Data_1:
|
|
|
|
case FK_Data_2:
|
|
|
|
case FK_Data_4:
|
|
|
|
case FK_Data_8:
|
|
|
|
return Value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
void AArch64AsmBackend::applyFixup(const MCFixup &Fixup, char *Data,
|
|
|
|
unsigned DataSize, uint64_t Value,
|
|
|
|
bool IsPCRel) const {
|
2014-03-29 18:18:08 +08:00
|
|
|
unsigned NumBytes = getFixupKindNumBytes(Fixup.getKind());
|
|
|
|
if (!Value)
|
|
|
|
return; // Doesn't change encoding.
|
|
|
|
MCFixupKindInfo Info = getFixupKindInfo(Fixup.getKind());
|
|
|
|
// Apply any target-specific value adjustments.
|
|
|
|
Value = adjustFixupValue(Fixup.getKind(), Value);
|
|
|
|
|
|
|
|
// Shift the value into position.
|
|
|
|
Value <<= Info.TargetOffset;
|
|
|
|
|
|
|
|
unsigned Offset = Fixup.getOffset();
|
|
|
|
assert(Offset + NumBytes <= DataSize && "Invalid fixup offset!");
|
|
|
|
|
|
|
|
// For each byte of the fragment that the fixup touches, mask in the
|
|
|
|
// bits from the fixup value.
|
|
|
|
for (unsigned i = 0; i != NumBytes; ++i)
|
|
|
|
Data[Offset + i] |= uint8_t((Value >> (i * 8)) & 0xff);
|
|
|
|
}
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
bool AArch64AsmBackend::mayNeedRelaxation(const MCInst &Inst) const {
|
2014-03-29 18:18:08 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
bool AArch64AsmBackend::fixupNeedsRelaxation(const MCFixup &Fixup,
|
|
|
|
uint64_t Value,
|
|
|
|
const MCRelaxableFragment *DF,
|
|
|
|
const MCAsmLayout &Layout) const {
|
|
|
|
// FIXME: This isn't correct for AArch64. Just moving the "generic" logic
|
2014-03-29 18:18:08 +08:00
|
|
|
// into the targets for now.
|
|
|
|
//
|
|
|
|
// Relax if the value is too big for a (signed) i8.
|
|
|
|
return int64_t(Value) != int64_t(int8_t(Value));
|
|
|
|
}
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
void AArch64AsmBackend::relaxInstruction(const MCInst &Inst,
|
|
|
|
MCInst &Res) const {
|
2015-01-05 18:15:49 +08:00
|
|
|
llvm_unreachable("AArch64AsmBackend::relaxInstruction() unimplemented");
|
2014-03-29 18:18:08 +08:00
|
|
|
}
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
bool AArch64AsmBackend::writeNopData(uint64_t Count, MCObjectWriter *OW) const {
|
2014-03-29 18:18:08 +08:00
|
|
|
// If the count is not 4-byte aligned, we must be writing data into the text
|
|
|
|
// section (otherwise we have unaligned instructions, and thus have far
|
|
|
|
// bigger problems), so just write zeros instead.
|
2015-04-17 19:12:43 +08:00
|
|
|
OW->WriteZeros(Count % 4);
|
2014-03-29 18:18:08 +08:00
|
|
|
|
|
|
|
// We are properly aligned, so write NOPs as requested.
|
|
|
|
Count /= 4;
|
|
|
|
for (uint64_t i = 0; i != Count; ++i)
|
2015-06-05 06:24:41 +08:00
|
|
|
OW->write32(0xd503201f);
|
2014-03-29 18:18:08 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
namespace CU {
|
|
|
|
|
|
|
|
/// \brief Compact unwind encoding values.
|
|
|
|
enum CompactUnwindEncodings {
|
|
|
|
/// \brief A "frameless" leaf function, where no non-volatile registers are
|
|
|
|
/// saved. The return remains in LR throughout the function.
|
2014-05-24 20:50:23 +08:00
|
|
|
UNWIND_AArch64_MODE_FRAMELESS = 0x02000000,
|
2014-03-29 18:18:08 +08:00
|
|
|
|
|
|
|
/// \brief No compact unwind encoding available. Instead the low 23-bits of
|
|
|
|
/// the compact unwind encoding is the offset of the DWARF FDE in the
|
|
|
|
/// __eh_frame section. This mode is never used in object files. It is only
|
|
|
|
/// generated by the linker in final linked images, which have only DWARF info
|
|
|
|
/// for a function.
|
2014-05-24 20:50:23 +08:00
|
|
|
UNWIND_AArch64_MODE_DWARF = 0x03000000,
|
2014-03-29 18:18:08 +08:00
|
|
|
|
|
|
|
/// \brief This is a standard arm64 prologue where FP/LR are immediately
|
|
|
|
/// pushed on the stack, then SP is copied to FP. If there are any
|
|
|
|
/// non-volatile register saved, they are copied into the stack fame in pairs
|
|
|
|
/// in a contiguous ranger right below the saved FP/LR pair. Any subset of the
|
|
|
|
/// five X pairs and four D pairs can be saved, but the memory layout must be
|
|
|
|
/// in register number order.
|
2014-05-24 20:50:23 +08:00
|
|
|
UNWIND_AArch64_MODE_FRAME = 0x04000000,
|
2014-03-29 18:18:08 +08:00
|
|
|
|
|
|
|
/// \brief Frame register pair encodings.
|
2014-05-24 20:50:23 +08:00
|
|
|
UNWIND_AArch64_FRAME_X19_X20_PAIR = 0x00000001,
|
|
|
|
UNWIND_AArch64_FRAME_X21_X22_PAIR = 0x00000002,
|
|
|
|
UNWIND_AArch64_FRAME_X23_X24_PAIR = 0x00000004,
|
|
|
|
UNWIND_AArch64_FRAME_X25_X26_PAIR = 0x00000008,
|
|
|
|
UNWIND_AArch64_FRAME_X27_X28_PAIR = 0x00000010,
|
|
|
|
UNWIND_AArch64_FRAME_D8_D9_PAIR = 0x00000100,
|
|
|
|
UNWIND_AArch64_FRAME_D10_D11_PAIR = 0x00000200,
|
|
|
|
UNWIND_AArch64_FRAME_D12_D13_PAIR = 0x00000400,
|
|
|
|
UNWIND_AArch64_FRAME_D14_D15_PAIR = 0x00000800
|
2014-03-29 18:18:08 +08:00
|
|
|
};
|
|
|
|
|
2015-06-23 17:49:53 +08:00
|
|
|
} // end CU namespace
|
2014-03-29 18:18:08 +08:00
|
|
|
|
|
|
|
// FIXME: This should be in a separate file.
|
2014-05-24 20:50:23 +08:00
|
|
|
class DarwinAArch64AsmBackend : public AArch64AsmBackend {
|
2014-03-29 18:18:08 +08:00
|
|
|
const MCRegisterInfo &MRI;
|
|
|
|
|
|
|
|
/// \brief Encode compact unwind stack adjustment for frameless functions.
|
2014-05-24 20:50:23 +08:00
|
|
|
/// See UNWIND_AArch64_FRAMELESS_STACK_SIZE_MASK in compact_unwind_encoding.h.
|
2014-03-29 18:18:08 +08:00
|
|
|
/// The stack size always needs to be 16 byte aligned.
|
|
|
|
uint32_t encodeStackAdjustment(uint32_t StackSize) const {
|
|
|
|
return (StackSize / 16) << 12;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
2014-05-24 20:50:23 +08:00
|
|
|
DarwinAArch64AsmBackend(const Target &T, const MCRegisterInfo &MRI)
|
|
|
|
: AArch64AsmBackend(T), MRI(MRI) {}
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2015-04-15 06:14:34 +08:00
|
|
|
MCObjectWriter *createObjectWriter(raw_pwrite_stream &OS) const override {
|
2014-05-24 20:50:23 +08:00
|
|
|
return createAArch64MachObjectWriter(OS, MachO::CPU_TYPE_ARM64,
|
|
|
|
MachO::CPU_SUBTYPE_ARM64_ALL);
|
2014-03-29 18:18:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Generate the compact unwind encoding from the CFI directives.
|
2014-04-29 15:58:25 +08:00
|
|
|
uint32_t generateCompactUnwindEncoding(
|
|
|
|
ArrayRef<MCCFIInstruction> Instrs) const override {
|
2014-03-29 18:18:08 +08:00
|
|
|
if (Instrs.empty())
|
2014-05-24 20:50:23 +08:00
|
|
|
return CU::UNWIND_AArch64_MODE_FRAMELESS;
|
2014-03-29 18:18:08 +08:00
|
|
|
|
|
|
|
bool HasFP = false;
|
|
|
|
unsigned StackSize = 0;
|
|
|
|
|
|
|
|
uint32_t CompactUnwindEncoding = 0;
|
|
|
|
for (size_t i = 0, e = Instrs.size(); i != e; ++i) {
|
|
|
|
const MCCFIInstruction &Inst = Instrs[i];
|
|
|
|
|
|
|
|
switch (Inst.getOperation()) {
|
|
|
|
default:
|
|
|
|
// Cannot handle this directive: bail out.
|
2014-05-24 20:50:23 +08:00
|
|
|
return CU::UNWIND_AArch64_MODE_DWARF;
|
2014-03-29 18:18:08 +08:00
|
|
|
case MCCFIInstruction::OpDefCfa: {
|
|
|
|
// Defines a frame pointer.
|
|
|
|
assert(getXRegFromWReg(MRI.getLLVMRegNum(Inst.getRegister(), true)) ==
|
2014-05-24 20:50:23 +08:00
|
|
|
AArch64::FP &&
|
2014-03-29 18:18:08 +08:00
|
|
|
"Invalid frame pointer!");
|
|
|
|
assert(i + 2 < e && "Insufficient CFI instructions to define a frame!");
|
|
|
|
|
|
|
|
const MCCFIInstruction &LRPush = Instrs[++i];
|
|
|
|
assert(LRPush.getOperation() == MCCFIInstruction::OpOffset &&
|
|
|
|
"Link register not pushed!");
|
|
|
|
const MCCFIInstruction &FPPush = Instrs[++i];
|
|
|
|
assert(FPPush.getOperation() == MCCFIInstruction::OpOffset &&
|
|
|
|
"Frame pointer not pushed!");
|
|
|
|
|
|
|
|
unsigned LRReg = MRI.getLLVMRegNum(LRPush.getRegister(), true);
|
|
|
|
unsigned FPReg = MRI.getLLVMRegNum(FPPush.getRegister(), true);
|
|
|
|
|
|
|
|
LRReg = getXRegFromWReg(LRReg);
|
|
|
|
FPReg = getXRegFromWReg(FPReg);
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
assert(LRReg == AArch64::LR && FPReg == AArch64::FP &&
|
2014-03-29 18:18:08 +08:00
|
|
|
"Pushing invalid registers for frame!");
|
|
|
|
|
|
|
|
// Indicate that the function has a frame.
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_MODE_FRAME;
|
2014-03-29 18:18:08 +08:00
|
|
|
HasFP = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case MCCFIInstruction::OpDefCfaOffset: {
|
|
|
|
assert(StackSize == 0 && "We already have the CFA offset!");
|
|
|
|
StackSize = std::abs(Inst.getOffset());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case MCCFIInstruction::OpOffset: {
|
|
|
|
// Registers are saved in pairs. We expect there to be two consecutive
|
|
|
|
// `.cfi_offset' instructions with the appropriate registers specified.
|
|
|
|
unsigned Reg1 = MRI.getLLVMRegNum(Inst.getRegister(), true);
|
|
|
|
if (i + 1 == e)
|
2014-05-24 20:50:23 +08:00
|
|
|
return CU::UNWIND_AArch64_MODE_DWARF;
|
2014-03-29 18:18:08 +08:00
|
|
|
|
|
|
|
const MCCFIInstruction &Inst2 = Instrs[++i];
|
|
|
|
if (Inst2.getOperation() != MCCFIInstruction::OpOffset)
|
2014-05-24 20:50:23 +08:00
|
|
|
return CU::UNWIND_AArch64_MODE_DWARF;
|
2014-03-29 18:18:08 +08:00
|
|
|
unsigned Reg2 = MRI.getLLVMRegNum(Inst2.getRegister(), true);
|
|
|
|
|
|
|
|
// N.B. The encodings must be in register number order, and the X
|
|
|
|
// registers before the D registers.
|
|
|
|
|
|
|
|
// X19/X20 pair = 0x00000001,
|
|
|
|
// X21/X22 pair = 0x00000002,
|
|
|
|
// X23/X24 pair = 0x00000004,
|
|
|
|
// X25/X26 pair = 0x00000008,
|
|
|
|
// X27/X28 pair = 0x00000010
|
|
|
|
Reg1 = getXRegFromWReg(Reg1);
|
|
|
|
Reg2 = getXRegFromWReg(Reg2);
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
if (Reg1 == AArch64::X19 && Reg2 == AArch64::X20 &&
|
2014-03-29 18:18:08 +08:00
|
|
|
(CompactUnwindEncoding & 0xF1E) == 0)
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_X19_X20_PAIR;
|
|
|
|
else if (Reg1 == AArch64::X21 && Reg2 == AArch64::X22 &&
|
2014-03-29 18:18:08 +08:00
|
|
|
(CompactUnwindEncoding & 0xF1C) == 0)
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_X21_X22_PAIR;
|
|
|
|
else if (Reg1 == AArch64::X23 && Reg2 == AArch64::X24 &&
|
2014-03-29 18:18:08 +08:00
|
|
|
(CompactUnwindEncoding & 0xF18) == 0)
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_X23_X24_PAIR;
|
|
|
|
else if (Reg1 == AArch64::X25 && Reg2 == AArch64::X26 &&
|
2014-03-29 18:18:08 +08:00
|
|
|
(CompactUnwindEncoding & 0xF10) == 0)
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_X25_X26_PAIR;
|
|
|
|
else if (Reg1 == AArch64::X27 && Reg2 == AArch64::X28 &&
|
2014-03-29 18:18:08 +08:00
|
|
|
(CompactUnwindEncoding & 0xF00) == 0)
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_X27_X28_PAIR;
|
2014-03-29 18:18:08 +08:00
|
|
|
else {
|
|
|
|
Reg1 = getDRegFromBReg(Reg1);
|
|
|
|
Reg2 = getDRegFromBReg(Reg2);
|
|
|
|
|
|
|
|
// D8/D9 pair = 0x00000100,
|
|
|
|
// D10/D11 pair = 0x00000200,
|
|
|
|
// D12/D13 pair = 0x00000400,
|
|
|
|
// D14/D15 pair = 0x00000800
|
2014-05-24 20:50:23 +08:00
|
|
|
if (Reg1 == AArch64::D8 && Reg2 == AArch64::D9 &&
|
2014-03-29 18:18:08 +08:00
|
|
|
(CompactUnwindEncoding & 0xE00) == 0)
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_D8_D9_PAIR;
|
|
|
|
else if (Reg1 == AArch64::D10 && Reg2 == AArch64::D11 &&
|
2014-03-29 18:18:08 +08:00
|
|
|
(CompactUnwindEncoding & 0xC00) == 0)
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_D10_D11_PAIR;
|
|
|
|
else if (Reg1 == AArch64::D12 && Reg2 == AArch64::D13 &&
|
2014-03-29 18:18:08 +08:00
|
|
|
(CompactUnwindEncoding & 0x800) == 0)
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_D12_D13_PAIR;
|
|
|
|
else if (Reg1 == AArch64::D14 && Reg2 == AArch64::D15)
|
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_FRAME_D14_D15_PAIR;
|
2014-03-29 18:18:08 +08:00
|
|
|
else
|
|
|
|
// A pair was pushed which we cannot handle.
|
2014-05-24 20:50:23 +08:00
|
|
|
return CU::UNWIND_AArch64_MODE_DWARF;
|
2014-03-29 18:18:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!HasFP) {
|
|
|
|
// With compact unwind info we can only represent stack adjustments of up
|
|
|
|
// to 65520 bytes.
|
|
|
|
if (StackSize > 65520)
|
2014-05-24 20:50:23 +08:00
|
|
|
return CU::UNWIND_AArch64_MODE_DWARF;
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
CompactUnwindEncoding |= CU::UNWIND_AArch64_MODE_FRAMELESS;
|
2014-03-29 18:18:08 +08:00
|
|
|
CompactUnwindEncoding |= encodeStackAdjustment(StackSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
return CompactUnwindEncoding;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
class ELFAArch64AsmBackend : public AArch64AsmBackend {
|
2014-03-29 18:18:08 +08:00
|
|
|
public:
|
|
|
|
uint8_t OSABI;
|
2014-04-23 18:26:40 +08:00
|
|
|
bool IsLittleEndian;
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
ELFAArch64AsmBackend(const Target &T, uint8_t OSABI, bool IsLittleEndian)
|
|
|
|
: AArch64AsmBackend(T), OSABI(OSABI), IsLittleEndian(IsLittleEndian) {}
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2015-04-15 06:14:34 +08:00
|
|
|
MCObjectWriter *createObjectWriter(raw_pwrite_stream &OS) const override {
|
2014-05-24 20:50:23 +08:00
|
|
|
return createAArch64ELFObjectWriter(OS, OSABI, IsLittleEndian);
|
2014-03-29 18:18:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void processFixupValue(const MCAssembler &Asm, const MCAsmLayout &Layout,
|
|
|
|
const MCFixup &Fixup, const MCFragment *DF,
|
|
|
|
const MCValue &Target, uint64_t &Value,
|
|
|
|
bool &IsResolved) override;
|
2014-05-15 00:51:58 +08:00
|
|
|
|
|
|
|
void applyFixup(const MCFixup &Fixup, char *Data, unsigned DataSize,
|
|
|
|
uint64_t Value, bool IsPCRel) const override;
|
2014-03-29 18:18:08 +08:00
|
|
|
};
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
void ELFAArch64AsmBackend::processFixupValue(
|
|
|
|
const MCAssembler &Asm, const MCAsmLayout &Layout, const MCFixup &Fixup,
|
|
|
|
const MCFragment *DF, const MCValue &Target, uint64_t &Value,
|
|
|
|
bool &IsResolved) {
|
2014-03-29 18:18:08 +08:00
|
|
|
// The ADRP instruction adds some multiple of 0x1000 to the current PC &
|
|
|
|
// ~0xfff. This means that the required offset to reach a symbol can vary by
|
|
|
|
// up to one step depending on where the ADRP is in memory. For example:
|
|
|
|
//
|
|
|
|
// ADRP x0, there
|
|
|
|
// there:
|
|
|
|
//
|
|
|
|
// If the ADRP occurs at address 0xffc then "there" will be at 0x1000 and
|
|
|
|
// we'll need that as an offset. At any other address "there" will be in the
|
|
|
|
// same page as the ADRP and the instruction should encode 0x0. Assuming the
|
|
|
|
// section isn't 0x1000-aligned, we therefore need to delegate this decision
|
|
|
|
// to the linker -- a relocation!
|
2014-05-24 20:50:23 +08:00
|
|
|
if ((uint32_t)Fixup.getKind() == AArch64::fixup_aarch64_pcrel_adrp_imm21)
|
2014-03-29 18:18:08 +08:00
|
|
|
IsResolved = false;
|
|
|
|
}
|
2014-05-15 00:51:58 +08:00
|
|
|
|
2015-03-25 05:47:03 +08:00
|
|
|
// Returns whether this fixup is based on an address in the .eh_frame section,
|
|
|
|
// and therefore should be byte swapped.
|
|
|
|
// FIXME: Should be replaced with something more principled.
|
|
|
|
static bool isByteSwappedFixup(const MCExpr *E) {
|
|
|
|
MCValue Val;
|
2015-05-30 09:25:56 +08:00
|
|
|
if (!E->evaluateAsRelocatable(Val, nullptr, nullptr))
|
2015-03-25 05:47:03 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!Val.getSymA() || Val.getSymA()->getSymbol().isUndefined())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const MCSectionELF *SecELF =
|
|
|
|
dyn_cast<MCSectionELF>(&Val.getSymA()->getSymbol().getSection());
|
|
|
|
return SecELF->getSectionName() == ".eh_frame";
|
|
|
|
}
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
void ELFAArch64AsmBackend::applyFixup(const MCFixup &Fixup, char *Data,
|
|
|
|
unsigned DataSize, uint64_t Value,
|
|
|
|
bool IsPCRel) const {
|
2014-05-15 00:51:58 +08:00
|
|
|
// store fixups in .eh_frame section in big endian order
|
|
|
|
if (!IsLittleEndian && Fixup.getKind() == FK_Data_4) {
|
2015-03-25 05:47:03 +08:00
|
|
|
if (isByteSwappedFixup(Fixup.getValue()))
|
2014-05-15 00:51:58 +08:00
|
|
|
Value = ByteSwap_32(unsigned(Value));
|
|
|
|
}
|
2014-05-24 20:50:23 +08:00
|
|
|
AArch64AsmBackend::applyFixup (Fixup, Data, DataSize, Value, IsPCRel);
|
2014-05-15 00:51:58 +08:00
|
|
|
}
|
2015-06-23 17:49:53 +08:00
|
|
|
}
|
2014-03-29 18:18:08 +08:00
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
MCAsmBackend *llvm::createAArch64leAsmBackend(const Target &T,
|
2015-06-10 18:35:34 +08:00
|
|
|
const MCRegisterInfo &MRI,
|
Re-commit r247683: Replace Triple with a new TargetTuple in MCTargetDesc/* and related. NFC.
Summary:
This is the first patch in the series to migrate Triple's (which are ambiguous)
to TargetTuple's (which aren't).
For the moment, TargetTuple simply passes all requests to the Triple object it
holds. Once it has replaced Triple, it will start to implement the interface in
a more suitable way.
This change makes some changes to the public C++ API. In particular,
InitMCSubtargetInfo(), createMCRelocationInfo(), and createMCSymbolizer()
now take TargetTuples instead of Triples. The other public C++ API's have
been left as-is for the moment to reduce patch size.
This commit also contains a trivial patch to clang to account for the C++ API
change. Thanks go to Pavel Labath for fixing LLDB for me.
Reviewers: rengolin
Subscribers: jyknight, dschuff, arsenm, rampitec, danalbert, srhines, javed.absar, dsanders, echristo, emaste, jholewinski, tberghammer, ted, jfb, llvm-commits, rengolin
Differential Revision: http://reviews.llvm.org/D10969
llvm-svn: 247692
2015-09-15 22:08:28 +08:00
|
|
|
const TargetTuple &TT,
|
2015-06-10 18:35:34 +08:00
|
|
|
StringRef CPU) {
|
Re-commit r247683: Replace Triple with a new TargetTuple in MCTargetDesc/* and related. NFC.
Summary:
This is the first patch in the series to migrate Triple's (which are ambiguous)
to TargetTuple's (which aren't).
For the moment, TargetTuple simply passes all requests to the Triple object it
holds. Once it has replaced Triple, it will start to implement the interface in
a more suitable way.
This change makes some changes to the public C++ API. In particular,
InitMCSubtargetInfo(), createMCRelocationInfo(), and createMCSymbolizer()
now take TargetTuples instead of Triples. The other public C++ API's have
been left as-is for the moment to reduce patch size.
This commit also contains a trivial patch to clang to account for the C++ API
change. Thanks go to Pavel Labath for fixing LLDB for me.
Reviewers: rengolin
Subscribers: jyknight, dschuff, arsenm, rampitec, danalbert, srhines, javed.absar, dsanders, echristo, emaste, jholewinski, tberghammer, ted, jfb, llvm-commits, rengolin
Differential Revision: http://reviews.llvm.org/D10969
llvm-svn: 247692
2015-09-15 22:08:28 +08:00
|
|
|
if (TT.isOSBinFormatMachO())
|
2014-05-24 20:50:23 +08:00
|
|
|
return new DarwinAArch64AsmBackend(T, MRI);
|
2014-03-29 18:18:08 +08:00
|
|
|
|
Re-commit r247683: Replace Triple with a new TargetTuple in MCTargetDesc/* and related. NFC.
Summary:
This is the first patch in the series to migrate Triple's (which are ambiguous)
to TargetTuple's (which aren't).
For the moment, TargetTuple simply passes all requests to the Triple object it
holds. Once it has replaced Triple, it will start to implement the interface in
a more suitable way.
This change makes some changes to the public C++ API. In particular,
InitMCSubtargetInfo(), createMCRelocationInfo(), and createMCSymbolizer()
now take TargetTuples instead of Triples. The other public C++ API's have
been left as-is for the moment to reduce patch size.
This commit also contains a trivial patch to clang to account for the C++ API
change. Thanks go to Pavel Labath for fixing LLDB for me.
Reviewers: rengolin
Subscribers: jyknight, dschuff, arsenm, rampitec, danalbert, srhines, javed.absar, dsanders, echristo, emaste, jholewinski, tberghammer, ted, jfb, llvm-commits, rengolin
Differential Revision: http://reviews.llvm.org/D10969
llvm-svn: 247692
2015-09-15 22:08:28 +08:00
|
|
|
assert(TT.isOSBinFormatELF() && "Expect either MachO or ELF target");
|
|
|
|
uint8_t OSABI = MCELFObjectTargetWriter::getOSABI(TT.getOS());
|
2014-08-07 00:05:02 +08:00
|
|
|
return new ELFAArch64AsmBackend(T, OSABI, /*IsLittleEndian=*/true);
|
2014-04-23 18:26:40 +08:00
|
|
|
}
|
|
|
|
|
2014-05-24 20:50:23 +08:00
|
|
|
MCAsmBackend *llvm::createAArch64beAsmBackend(const Target &T,
|
2015-06-10 18:35:34 +08:00
|
|
|
const MCRegisterInfo &MRI,
|
Re-commit r247683: Replace Triple with a new TargetTuple in MCTargetDesc/* and related. NFC.
Summary:
This is the first patch in the series to migrate Triple's (which are ambiguous)
to TargetTuple's (which aren't).
For the moment, TargetTuple simply passes all requests to the Triple object it
holds. Once it has replaced Triple, it will start to implement the interface in
a more suitable way.
This change makes some changes to the public C++ API. In particular,
InitMCSubtargetInfo(), createMCRelocationInfo(), and createMCSymbolizer()
now take TargetTuples instead of Triples. The other public C++ API's have
been left as-is for the moment to reduce patch size.
This commit also contains a trivial patch to clang to account for the C++ API
change. Thanks go to Pavel Labath for fixing LLDB for me.
Reviewers: rengolin
Subscribers: jyknight, dschuff, arsenm, rampitec, danalbert, srhines, javed.absar, dsanders, echristo, emaste, jholewinski, tberghammer, ted, jfb, llvm-commits, rengolin
Differential Revision: http://reviews.llvm.org/D10969
llvm-svn: 247692
2015-09-15 22:08:28 +08:00
|
|
|
const TargetTuple &TT,
|
2015-06-10 18:35:34 +08:00
|
|
|
StringRef CPU) {
|
Re-commit r247683: Replace Triple with a new TargetTuple in MCTargetDesc/* and related. NFC.
Summary:
This is the first patch in the series to migrate Triple's (which are ambiguous)
to TargetTuple's (which aren't).
For the moment, TargetTuple simply passes all requests to the Triple object it
holds. Once it has replaced Triple, it will start to implement the interface in
a more suitable way.
This change makes some changes to the public C++ API. In particular,
InitMCSubtargetInfo(), createMCRelocationInfo(), and createMCSymbolizer()
now take TargetTuples instead of Triples. The other public C++ API's have
been left as-is for the moment to reduce patch size.
This commit also contains a trivial patch to clang to account for the C++ API
change. Thanks go to Pavel Labath for fixing LLDB for me.
Reviewers: rengolin
Subscribers: jyknight, dschuff, arsenm, rampitec, danalbert, srhines, javed.absar, dsanders, echristo, emaste, jholewinski, tberghammer, ted, jfb, llvm-commits, rengolin
Differential Revision: http://reviews.llvm.org/D10969
llvm-svn: 247692
2015-09-15 22:08:28 +08:00
|
|
|
assert(TT.isOSBinFormatELF() &&
|
2014-05-24 20:50:23 +08:00
|
|
|
"Big endian is only supported for ELF targets!");
|
Re-commit r247683: Replace Triple with a new TargetTuple in MCTargetDesc/* and related. NFC.
Summary:
This is the first patch in the series to migrate Triple's (which are ambiguous)
to TargetTuple's (which aren't).
For the moment, TargetTuple simply passes all requests to the Triple object it
holds. Once it has replaced Triple, it will start to implement the interface in
a more suitable way.
This change makes some changes to the public C++ API. In particular,
InitMCSubtargetInfo(), createMCRelocationInfo(), and createMCSymbolizer()
now take TargetTuples instead of Triples. The other public C++ API's have
been left as-is for the moment to reduce patch size.
This commit also contains a trivial patch to clang to account for the C++ API
change. Thanks go to Pavel Labath for fixing LLDB for me.
Reviewers: rengolin
Subscribers: jyknight, dschuff, arsenm, rampitec, danalbert, srhines, javed.absar, dsanders, echristo, emaste, jholewinski, tberghammer, ted, jfb, llvm-commits, rengolin
Differential Revision: http://reviews.llvm.org/D10969
llvm-svn: 247692
2015-09-15 22:08:28 +08:00
|
|
|
uint8_t OSABI = MCELFObjectTargetWriter::getOSABI(TT.getOS());
|
2014-08-07 00:05:02 +08:00
|
|
|
return new ELFAArch64AsmBackend(T, OSABI,
|
2014-05-24 20:50:23 +08:00
|
|
|
/*IsLittleEndian=*/false);
|
2014-03-29 18:18:08 +08:00
|
|
|
}
|