2010-04-09 00:30:25 +08:00
|
|
|
//===--- CGVTables.h - Emit LLVM Code for C++ vtables ---------------------===//
|
2009-10-12 06:13:54 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This contains code dealing with C++ code generation of virtual tables.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef CLANG_CODEGEN_CGVTABLE_H
|
|
|
|
#define CLANG_CODEGEN_CGVTABLE_H
|
|
|
|
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2009-12-06 08:23:49 +08:00
|
|
|
#include "llvm/GlobalVariable.h"
|
2009-11-14 01:08:56 +08:00
|
|
|
#include "GlobalDecl.h"
|
2009-10-12 06:13:54 +08:00
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
class CXXRecordDecl;
|
2009-11-26 21:09:03 +08:00
|
|
|
|
2009-10-12 06:13:54 +08:00
|
|
|
namespace CodeGen {
|
|
|
|
class CodeGenModule;
|
2009-11-26 10:32:05 +08:00
|
|
|
|
2010-03-23 23:13:06 +08:00
|
|
|
/// ReturnAdjustment - A return adjustment.
|
|
|
|
struct ReturnAdjustment {
|
|
|
|
/// NonVirtual - The non-virtual adjustment from the derived object to its
|
|
|
|
/// nearest virtual base.
|
|
|
|
int64_t NonVirtual;
|
|
|
|
|
|
|
|
/// VBaseOffsetOffset - The offset (in bytes), relative to the address point
|
|
|
|
/// of the virtual base class offset.
|
|
|
|
int64_t VBaseOffsetOffset;
|
|
|
|
|
|
|
|
ReturnAdjustment() : NonVirtual(0), VBaseOffsetOffset(0) { }
|
|
|
|
|
|
|
|
bool isEmpty() const { return !NonVirtual && !VBaseOffsetOffset; }
|
|
|
|
|
|
|
|
friend bool operator==(const ReturnAdjustment &LHS,
|
|
|
|
const ReturnAdjustment &RHS) {
|
|
|
|
return LHS.NonVirtual == RHS.NonVirtual &&
|
|
|
|
LHS.VBaseOffsetOffset == RHS.VBaseOffsetOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const ReturnAdjustment &LHS,
|
|
|
|
const ReturnAdjustment &RHS) {
|
|
|
|
if (LHS.NonVirtual < RHS.NonVirtual)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return LHS.NonVirtual == RHS.NonVirtual &&
|
|
|
|
LHS.VBaseOffsetOffset < RHS.VBaseOffsetOffset;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/// ThisAdjustment - A 'this' pointer adjustment.
|
|
|
|
struct ThisAdjustment {
|
|
|
|
/// NonVirtual - The non-virtual adjustment from the derived object to its
|
|
|
|
/// nearest virtual base.
|
|
|
|
int64_t NonVirtual;
|
|
|
|
|
|
|
|
/// VCallOffsetOffset - The offset (in bytes), relative to the address point,
|
|
|
|
/// of the virtual call offset.
|
|
|
|
int64_t VCallOffsetOffset;
|
|
|
|
|
|
|
|
ThisAdjustment() : NonVirtual(0), VCallOffsetOffset(0) { }
|
|
|
|
|
|
|
|
bool isEmpty() const { return !NonVirtual && !VCallOffsetOffset; }
|
|
|
|
|
|
|
|
friend bool operator==(const ThisAdjustment &LHS,
|
|
|
|
const ThisAdjustment &RHS) {
|
|
|
|
return LHS.NonVirtual == RHS.NonVirtual &&
|
|
|
|
LHS.VCallOffsetOffset == RHS.VCallOffsetOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const ThisAdjustment &LHS,
|
|
|
|
const ThisAdjustment &RHS) {
|
|
|
|
if (LHS.NonVirtual < RHS.NonVirtual)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return LHS.NonVirtual == RHS.NonVirtual &&
|
|
|
|
LHS.VCallOffsetOffset < RHS.VCallOffsetOffset;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-03-23 23:17:13 +08:00
|
|
|
/// ThunkInfo - The 'this' pointer adjustment as well as an optional return
|
|
|
|
/// adjustment for a thunk.
|
|
|
|
struct ThunkInfo {
|
|
|
|
/// This - The 'this' pointer adjustment.
|
|
|
|
ThisAdjustment This;
|
|
|
|
|
|
|
|
/// Return - The return adjustment.
|
|
|
|
ReturnAdjustment Return;
|
|
|
|
|
|
|
|
ThunkInfo() { }
|
|
|
|
|
|
|
|
ThunkInfo(const ThisAdjustment &This, const ReturnAdjustment &Return)
|
|
|
|
: This(This), Return(Return) { }
|
|
|
|
|
|
|
|
friend bool operator==(const ThunkInfo &LHS, const ThunkInfo &RHS) {
|
|
|
|
return LHS.This == RHS.This && LHS.Return == RHS.Return;
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator<(const ThunkInfo &LHS, const ThunkInfo &RHS) {
|
|
|
|
if (LHS.This < RHS.This)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return LHS.This == RHS.This && LHS.Return < RHS.Return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isEmpty() const { return This.isEmpty() && Return.isEmpty(); }
|
|
|
|
};
|
|
|
|
|
2010-01-14 04:11:15 +08:00
|
|
|
// BaseSubobject - Uniquely identifies a direct or indirect base class.
|
|
|
|
// Stores both the base class decl and the offset from the most derived class to
|
|
|
|
// the base class.
|
|
|
|
class BaseSubobject {
|
|
|
|
/// Base - The base class declaration.
|
|
|
|
const CXXRecordDecl *Base;
|
|
|
|
|
|
|
|
/// BaseOffset - The offset from the most derived class to the base class.
|
|
|
|
uint64_t BaseOffset;
|
|
|
|
|
|
|
|
public:
|
|
|
|
BaseSubobject(const CXXRecordDecl *Base, uint64_t BaseOffset)
|
|
|
|
: Base(Base), BaseOffset(BaseOffset) { }
|
|
|
|
|
|
|
|
/// getBase - Returns the base class declaration.
|
|
|
|
const CXXRecordDecl *getBase() const { return Base; }
|
|
|
|
|
|
|
|
/// getBaseOffset - Returns the base class offset.
|
|
|
|
uint64_t getBaseOffset() const { return BaseOffset; }
|
2010-01-14 09:39:42 +08:00
|
|
|
|
2010-01-14 04:11:15 +08:00
|
|
|
friend bool operator==(const BaseSubobject &LHS, const BaseSubobject &RHS) {
|
|
|
|
return LHS.Base == RHS.Base && LHS.BaseOffset == RHS.BaseOffset;
|
|
|
|
}
|
|
|
|
};
|
2010-01-14 09:39:42 +08:00
|
|
|
|
|
|
|
} // end namespace CodeGen
|
|
|
|
} // end namespace clang
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
|
|
|
template<> struct DenseMapInfo<clang::CodeGen::BaseSubobject> {
|
|
|
|
static clang::CodeGen::BaseSubobject getEmptyKey() {
|
|
|
|
return clang::CodeGen::BaseSubobject(
|
|
|
|
DenseMapInfo<const clang::CXXRecordDecl *>::getEmptyKey(),
|
|
|
|
DenseMapInfo<uint64_t>::getEmptyKey());
|
|
|
|
}
|
|
|
|
|
|
|
|
static clang::CodeGen::BaseSubobject getTombstoneKey() {
|
|
|
|
return clang::CodeGen::BaseSubobject(
|
|
|
|
DenseMapInfo<const clang::CXXRecordDecl *>::getTombstoneKey(),
|
|
|
|
DenseMapInfo<uint64_t>::getTombstoneKey());
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned getHashValue(const clang::CodeGen::BaseSubobject &Base) {
|
|
|
|
return
|
|
|
|
DenseMapInfo<const clang::CXXRecordDecl *>::getHashValue(Base.getBase()) ^
|
|
|
|
DenseMapInfo<uint64_t>::getHashValue(Base.getBaseOffset());
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool isEqual(const clang::CodeGen::BaseSubobject &LHS,
|
|
|
|
const clang::CodeGen::BaseSubobject &RHS) {
|
|
|
|
return LHS == RHS;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-01-14 10:29:07 +08:00
|
|
|
// It's OK to treat BaseSubobject as a POD type.
|
|
|
|
template <> struct isPodLike<clang::CodeGen::BaseSubobject> {
|
|
|
|
static const bool value = true;
|
|
|
|
};
|
|
|
|
|
2010-01-14 09:39:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace CodeGen {
|
|
|
|
|
2010-03-23 12:11:45 +08:00
|
|
|
class CodeGenVTables {
|
2009-10-12 06:13:54 +08:00
|
|
|
CodeGenModule &CGM;
|
2009-11-26 21:09:03 +08:00
|
|
|
|
2010-04-18 04:15:18 +08:00
|
|
|
/// MethodVTableIndices - Contains the index (relative to the vtable address
|
2009-10-12 06:13:54 +08:00
|
|
|
/// point) where the function pointer for a virtual function is stored.
|
2010-04-18 04:15:18 +08:00
|
|
|
typedef llvm::DenseMap<GlobalDecl, int64_t> MethodVTableIndicesTy;
|
|
|
|
MethodVTableIndicesTy MethodVTableIndices;
|
2009-11-26 21:09:03 +08:00
|
|
|
|
2009-10-12 06:13:54 +08:00
|
|
|
typedef std::pair<const CXXRecordDecl *,
|
|
|
|
const CXXRecordDecl *> ClassPairTy;
|
2009-11-26 21:09:03 +08:00
|
|
|
|
2010-03-11 15:15:17 +08:00
|
|
|
/// VirtualBaseClassOffsetOffsets - Contains the vtable offset (relative to
|
|
|
|
/// the address point) in bytes where the offsets for virtual bases of a class
|
|
|
|
/// are stored.
|
|
|
|
typedef llvm::DenseMap<ClassPairTy, int64_t>
|
|
|
|
VirtualBaseClassOffsetOffsetsMapTy;
|
|
|
|
VirtualBaseClassOffsetOffsetsMapTy VirtualBaseClassOffsetOffsets;
|
2009-11-10 15:44:33 +08:00
|
|
|
|
2010-04-18 04:15:18 +08:00
|
|
|
/// VTables - All the vtables which have been defined.
|
|
|
|
llvm::DenseMap<const CXXRecordDecl *, llvm::GlobalVariable *> VTables;
|
2009-11-28 04:47:55 +08:00
|
|
|
|
|
|
|
/// NumVirtualFunctionPointers - Contains the number of virtual function
|
|
|
|
/// pointers in the vtable for a given record decl.
|
|
|
|
llvm::DenseMap<const CXXRecordDecl *, uint64_t> NumVirtualFunctionPointers;
|
|
|
|
|
2010-03-24 00:36:50 +08:00
|
|
|
typedef llvm::SmallVector<ThunkInfo, 1> ThunkInfoVectorTy;
|
|
|
|
typedef llvm::DenseMap<const CXXMethodDecl *, ThunkInfoVectorTy> ThunksMapTy;
|
|
|
|
|
|
|
|
/// Thunks - Contains all thunks that a given method decl will need.
|
|
|
|
ThunksMapTy Thunks;
|
|
|
|
|
2010-03-25 00:42:11 +08:00
|
|
|
typedef llvm::DenseMap<const CXXRecordDecl *, uint64_t *> VTableLayoutMapTy;
|
|
|
|
|
|
|
|
/// VTableLayoutMap - Stores the vtable layout for all record decls.
|
|
|
|
/// The layout is stored as an array of 64-bit integers, where the first
|
|
|
|
/// integer is the number of vtable entries in the layout, and the subsequent
|
|
|
|
/// integers are the vtable components.
|
|
|
|
VTableLayoutMapTy VTableLayoutMap;
|
|
|
|
|
2010-03-25 08:35:49 +08:00
|
|
|
typedef llvm::DenseMap<std::pair<const CXXRecordDecl *,
|
|
|
|
BaseSubobject>, uint64_t> AddressPointsMapTy;
|
2010-03-25 08:51:13 +08:00
|
|
|
|
2010-03-26 11:56:54 +08:00
|
|
|
/// Address points - Address points for all vtables.
|
2010-03-25 08:51:13 +08:00
|
|
|
AddressPointsMapTy AddressPoints;
|
2010-03-26 11:56:54 +08:00
|
|
|
|
|
|
|
/// VTableAddressPointsMapTy - Address points for a single vtable.
|
|
|
|
typedef llvm::DenseMap<BaseSubobject, uint64_t> VTableAddressPointsMapTy;
|
|
|
|
|
2010-03-25 23:26:28 +08:00
|
|
|
typedef llvm::SmallVector<std::pair<uint64_t, ThunkInfo>, 1>
|
|
|
|
VTableThunksTy;
|
|
|
|
|
|
|
|
typedef llvm::DenseMap<const CXXRecordDecl *, VTableThunksTy>
|
|
|
|
VTableThunksMapTy;
|
|
|
|
|
|
|
|
/// VTableThunksMap - Contains thunks needed by vtables.
|
|
|
|
VTableThunksMapTy VTableThunksMap;
|
|
|
|
|
2010-03-25 00:42:11 +08:00
|
|
|
uint64_t getNumVTableComponents(const CXXRecordDecl *RD) const {
|
|
|
|
assert(VTableLayoutMap.count(RD) && "No vtable layout for this class!");
|
|
|
|
|
|
|
|
return VTableLayoutMap.lookup(RD)[0];
|
|
|
|
}
|
|
|
|
|
2010-03-29 11:38:52 +08:00
|
|
|
const uint64_t *getVTableComponentsData(const CXXRecordDecl *RD) const {
|
|
|
|
assert(VTableLayoutMap.count(RD) && "No vtable layout for this class!");
|
|
|
|
|
|
|
|
uint64_t *Components = VTableLayoutMap.lookup(RD);
|
|
|
|
return &Components[1];
|
|
|
|
}
|
|
|
|
|
2010-03-26 12:23:58 +08:00
|
|
|
typedef llvm::DenseMap<ClassPairTy, uint64_t> SubVTTIndiciesMapTy;
|
|
|
|
|
|
|
|
/// SubVTTIndicies - Contains indices into the various sub-VTTs.
|
|
|
|
SubVTTIndiciesMapTy SubVTTIndicies;
|
|
|
|
|
|
|
|
|
|
|
|
typedef llvm::DenseMap<std::pair<const CXXRecordDecl *,
|
|
|
|
BaseSubobject>, uint64_t>
|
|
|
|
SecondaryVirtualPointerIndicesMapTy;
|
|
|
|
|
|
|
|
/// SecondaryVirtualPointerIndices - Contains the secondary virtual pointer
|
|
|
|
/// indices.
|
|
|
|
SecondaryVirtualPointerIndicesMapTy SecondaryVirtualPointerIndices;
|
2010-01-02 09:01:18 +08:00
|
|
|
|
2009-11-28 04:47:55 +08:00
|
|
|
/// getNumVirtualFunctionPointers - Return the number of virtual function
|
|
|
|
/// pointers in the vtable for a given record decl.
|
|
|
|
uint64_t getNumVirtualFunctionPointers(const CXXRecordDecl *RD);
|
|
|
|
|
2010-04-18 04:15:18 +08:00
|
|
|
void ComputeMethodVTableIndices(const CXXRecordDecl *RD);
|
2009-12-06 09:09:21 +08:00
|
|
|
|
|
|
|
llvm::GlobalVariable *GenerateVTT(llvm::GlobalVariable::LinkageTypes Linkage,
|
2010-01-02 09:01:18 +08:00
|
|
|
bool GenerateDefinition,
|
2009-12-06 09:09:21 +08:00
|
|
|
const CXXRecordDecl *RD);
|
|
|
|
|
2010-03-24 00:36:50 +08:00
|
|
|
/// EmitThunk - Emit a single thunk.
|
|
|
|
void EmitThunk(GlobalDecl GD, const ThunkInfo &Thunk);
|
|
|
|
|
2010-03-23 12:59:02 +08:00
|
|
|
/// EmitThunks - Emit the associated thunks for the given global decl.
|
|
|
|
void EmitThunks(GlobalDecl GD);
|
|
|
|
|
2010-03-25 00:42:11 +08:00
|
|
|
/// ComputeVTableRelatedInformation - Compute and store all vtable related
|
|
|
|
/// information (vtable layout, vbase offset offsets, thunks etc) for the
|
|
|
|
/// given record decl.
|
|
|
|
void ComputeVTableRelatedInformation(const CXXRecordDecl *RD);
|
|
|
|
|
2010-03-25 23:26:28 +08:00
|
|
|
/// CreateVTableInitializer - Create a vtable initializer for the given record
|
|
|
|
/// decl.
|
|
|
|
/// \param Components - The vtable components; this is really an array of
|
|
|
|
/// VTableComponents.
|
|
|
|
llvm::Constant *CreateVTableInitializer(const CXXRecordDecl *RD,
|
|
|
|
const uint64_t *Components,
|
|
|
|
unsigned NumComponents,
|
|
|
|
const VTableThunksTy &VTableThunks);
|
2010-03-26 11:56:54 +08:00
|
|
|
|
2009-10-12 06:13:54 +08:00
|
|
|
public:
|
2010-03-23 12:11:45 +08:00
|
|
|
CodeGenVTables(CodeGenModule &CGM)
|
2009-10-12 06:13:54 +08:00
|
|
|
: CGM(CGM) { }
|
|
|
|
|
2010-04-19 08:44:22 +08:00
|
|
|
// isKeyFunctionInAnotherTU - True if this record has a key function and it is
|
|
|
|
// in another translation unit.
|
|
|
|
static bool isKeyFunctionInAnotherTU(ASTContext &Context,
|
|
|
|
const CXXRecordDecl *RD) {
|
|
|
|
assert (RD->isDynamicClass() && "Non dynamic classes have no key.");
|
|
|
|
const CXXMethodDecl *KeyFunction = Context.getKeyFunction(RD);
|
|
|
|
return KeyFunction && !KeyFunction->getBody();
|
|
|
|
}
|
|
|
|
|
2010-01-02 09:01:18 +08:00
|
|
|
/// needsVTTParameter - Return whether the given global decl needs a VTT
|
|
|
|
/// parameter, which it does if it's a base constructor or destructor with
|
|
|
|
/// virtual bases.
|
|
|
|
static bool needsVTTParameter(GlobalDecl GD);
|
|
|
|
|
|
|
|
/// getSubVTTIndex - Return the index of the sub-VTT for the base class of the
|
|
|
|
/// given record decl.
|
|
|
|
uint64_t getSubVTTIndex(const CXXRecordDecl *RD, const CXXRecordDecl *Base);
|
|
|
|
|
2010-03-26 12:23:58 +08:00
|
|
|
/// getSecondaryVirtualPointerIndex - Return the index in the VTT where the
|
|
|
|
/// virtual pointer for the given subobject is located.
|
|
|
|
uint64_t getSecondaryVirtualPointerIndex(const CXXRecordDecl *RD,
|
|
|
|
BaseSubobject Base);
|
|
|
|
|
2010-04-18 04:15:18 +08:00
|
|
|
/// getMethodVTableIndex - Return the index (relative to the vtable address
|
2009-11-26 21:09:03 +08:00
|
|
|
/// point) where the function pointer for the given virtual function is
|
2009-10-12 06:13:54 +08:00
|
|
|
/// stored.
|
2010-04-18 04:15:18 +08:00
|
|
|
uint64_t getMethodVTableIndex(GlobalDecl GD);
|
2009-11-26 21:09:03 +08:00
|
|
|
|
2010-03-11 15:15:17 +08:00
|
|
|
/// getVirtualBaseOffsetOffset - Return the offset in bytes (relative to the
|
|
|
|
/// vtable address point) where the offset of the virtual base that contains
|
|
|
|
/// the given base is stored, otherwise, if no virtual base contains the given
|
2009-10-14 06:54:56 +08:00
|
|
|
/// class, return 0. Base must be a virtual base class or an unambigious
|
|
|
|
/// base.
|
2010-03-11 15:15:17 +08:00
|
|
|
int64_t getVirtualBaseOffsetOffset(const CXXRecordDecl *RD,
|
|
|
|
const CXXRecordDecl *VBase);
|
2009-11-10 15:44:33 +08:00
|
|
|
|
2010-03-29 10:08:26 +08:00
|
|
|
/// getAddressPoint - Get the address point of the given subobject in the
|
|
|
|
/// class decl.
|
|
|
|
uint64_t getAddressPoint(BaseSubobject Base, const CXXRecordDecl *RD);
|
|
|
|
|
2010-03-24 13:32:05 +08:00
|
|
|
/// GetAddrOfVTable - Get the address of the vtable for the given record decl.
|
2010-03-30 11:35:35 +08:00
|
|
|
llvm::GlobalVariable *GetAddrOfVTable(const CXXRecordDecl *RD);
|
2010-03-24 11:57:14 +08:00
|
|
|
|
2010-03-29 11:38:52 +08:00
|
|
|
/// EmitVTableDefinition - Emit the definition of the given vtable.
|
|
|
|
void EmitVTableDefinition(llvm::GlobalVariable *VTable,
|
|
|
|
llvm::GlobalVariable::LinkageTypes Linkage,
|
|
|
|
const CXXRecordDecl *RD);
|
|
|
|
|
2010-03-25 08:35:49 +08:00
|
|
|
/// GenerateConstructionVTable - Generate a construction vtable for the given
|
|
|
|
/// base subobject.
|
|
|
|
llvm::GlobalVariable *
|
|
|
|
GenerateConstructionVTable(const CXXRecordDecl *RD, const BaseSubobject &Base,
|
|
|
|
bool BaseIsVirtual,
|
2010-03-26 11:56:54 +08:00
|
|
|
VTableAddressPointsMapTy& AddressPoints);
|
2009-12-01 07:41:22 +08:00
|
|
|
|
2010-01-02 09:01:18 +08:00
|
|
|
llvm::GlobalVariable *getVTT(const CXXRecordDecl *RD);
|
2009-12-01 07:41:22 +08:00
|
|
|
|
2010-03-23 12:15:00 +08:00
|
|
|
// EmitVTableRelatedData - Will emit any thunks that the global decl might
|
|
|
|
// have, as well as the vtable itself if the global decl is the key function.
|
|
|
|
void EmitVTableRelatedData(GlobalDecl GD);
|
2010-03-10 10:19:29 +08:00
|
|
|
|
2010-03-24 02:18:41 +08:00
|
|
|
/// GenerateClassData - Generate all the class data required to be generated
|
2010-03-10 10:19:29 +08:00
|
|
|
/// upon definition of a KeyFunction. This includes the vtable, the
|
|
|
|
/// rtti data structure and the VTT.
|
|
|
|
///
|
|
|
|
/// \param Linkage - The desired linkage of the vtable, the RTTI and the VTT.
|
|
|
|
void GenerateClassData(llvm::GlobalVariable::LinkageTypes Linkage,
|
|
|
|
const CXXRecordDecl *RD);
|
2009-10-12 06:13:54 +08:00
|
|
|
};
|
2009-11-26 21:09:03 +08:00
|
|
|
|
2010-01-14 10:29:07 +08:00
|
|
|
} // end namespace CodeGen
|
|
|
|
} // end namespace clang
|
2009-10-12 06:13:54 +08:00
|
|
|
#endif
|