llvm-project/llvm/lib/Support/CSKYAttributeParser.cpp
Zi Xuan Wu 21bce9007a [Support] Add CSKY target parser and attributes parser
Construct LLVM Support module about CSKY target parser and attribute parser.
It refers CSKY ABIv2 and implementation of GNU binutils and GCC.

https://github.com/c-sky/csky-doc/blob/master/C-SKY_V2_CPU_Applications_Binary_Interface_Standards_Manual.pdf

Now we only support CSKY 800 series cpus and newer cpus in the future undering CSKYv2 ABI specification.
There are 11 archs including ck801, ck802, ck803, ck803s, ck804, ck805, ck807, ck810, ck810v, ck860, ck860v.

Every arch has base extensions, the cpus of that arch family have more extended extensions than base extensions.
We need specify extended extensions for every cpu. Every extension has its enum value, name and related llvm feature string with +/-.
Every enum value represents a bit of uint64_t integer.

Differential Revision: https://reviews.llvm.org/D119917
2022-02-28 11:35:07 +08:00

156 lines
4.7 KiB
C++

//===-- CSKYAttributeParser.cpp - CSKY Attribute Parser -----------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "llvm/Support/CSKYAttributeParser.h"
#include "llvm/ADT/StringExtras.h"
#include "llvm/Support/Errc.h"
using namespace llvm;
const CSKYAttributeParser::DisplayHandler
CSKYAttributeParser::displayRoutines[] = {
{
CSKYAttrs::CSKY_ARCH_NAME,
&ELFAttributeParser::stringAttribute,
},
{
CSKYAttrs::CSKY_CPU_NAME,
&ELFAttributeParser::stringAttribute,
},
{
CSKYAttrs::CSKY_ISA_FLAGS,
&ELFAttributeParser::integerAttribute,
},
{
CSKYAttrs::CSKY_ISA_EXT_FLAGS,
&ELFAttributeParser::integerAttribute,
},
{
CSKYAttrs::CSKY_DSP_VERSION,
&CSKYAttributeParser::dspVersion,
},
{
CSKYAttrs::CSKY_VDSP_VERSION,
&CSKYAttributeParser::vdspVersion,
},
{
CSKYAttrs::CSKY_FPU_VERSION,
&CSKYAttributeParser::fpuVersion,
},
{
CSKYAttrs::CSKY_FPU_ABI,
&CSKYAttributeParser::fpuABI,
},
{
CSKYAttrs::CSKY_FPU_ROUNDING,
&CSKYAttributeParser::fpuRounding,
},
{
CSKYAttrs::CSKY_FPU_DENORMAL,
&CSKYAttributeParser::fpuDenormal,
},
{
CSKYAttrs::CSKY_FPU_EXCEPTION,
&CSKYAttributeParser::fpuException,
},
{
CSKYAttrs::CSKY_FPU_NUMBER_MODULE,
&ELFAttributeParser::stringAttribute,
},
{
CSKYAttrs::CSKY_FPU_HARDFP,
&CSKYAttributeParser::fpuHardFP,
}};
Error CSKYAttributeParser::handler(uint64_t tag, bool &handled) {
handled = false;
for (unsigned AHI = 0, AHE = array_lengthof(displayRoutines); AHI != AHE;
++AHI) {
if (uint64_t(displayRoutines[AHI].attribute) == tag) {
if (Error e = (this->*displayRoutines[AHI].routine)(tag))
return e;
handled = true;
break;
}
}
return Error::success();
}
Error CSKYAttributeParser::dspVersion(unsigned tag) {
static const char *strings[] = {"Error", "DSP Extension", "DSP 2.0"};
return parseStringAttribute("Tag_CSKY_DSP_VERSION", tag,
makeArrayRef(strings));
}
Error CSKYAttributeParser::vdspVersion(unsigned tag) {
static const char *strings[] = {"Error", "VDSP Version 1", "VDSP Version 2"};
return parseStringAttribute("Tag_CSKY_VDSP_VERSION", tag,
makeArrayRef(strings));
}
Error CSKYAttributeParser::fpuVersion(unsigned tag) {
static const char *strings[] = {"Error", "FPU Version 1", "FPU Version 2",
"FPU Version 3"};
return parseStringAttribute("Tag_CSKY_FPU_VERSION", tag,
makeArrayRef(strings));
}
Error CSKYAttributeParser::fpuABI(unsigned tag) {
static const char *strings[] = {"Error", "Soft", "SoftFP", "Hard"};
return parseStringAttribute("Tag_CSKY_FPU_ABI", tag, makeArrayRef(strings));
}
Error CSKYAttributeParser::fpuRounding(unsigned tag) {
static const char *strings[] = {"None", "Needed"};
return parseStringAttribute("Tag_CSKY_FPU_ROUNDING", tag,
makeArrayRef(strings));
}
Error CSKYAttributeParser::fpuDenormal(unsigned tag) {
static const char *strings[] = {"None", "Needed"};
return parseStringAttribute("Tag_CSKY_FPU_DENORMAL", tag,
makeArrayRef(strings));
}
Error CSKYAttributeParser::fpuException(unsigned tag) {
static const char *strings[] = {"None", "Needed"};
return parseStringAttribute("Tag_CSKY_FPU_EXCEPTION", tag,
makeArrayRef(strings));
}
Error CSKYAttributeParser::fpuHardFP(unsigned tag) {
uint64_t value = de.getULEB128(cursor);
ListSeparator LS(" ");
std::string description;
if (value & 0x1) {
description += LS;
description += "Half";
}
if ((value >> 1) & 0x1) {
description += LS;
description += "Single";
}
if ((value >> 2) & 0x1) {
description += LS;
description += "Double";
}
if (description.empty()) {
printAttribute(tag, value, "");
return createStringError(errc::invalid_argument,
"unknown Tag_CSKY_FPU_HARDFP value: " +
Twine(value));
}
printAttribute(tag, value, description);
return Error::success();
}