// 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.
package clang
import (
"path/filepath"
"strings"
"android/soong/android"
"android/soong/genrule"
"github.com/google/blueprint"
)
func init() {
android.RegisterModuleType("clang_tblgen", clangTblgenFactory)
}
var (
pctx = android.NewPackageContext("android/soong/clang")
clangTblgen = pctx.HostBinToolVariable("clangTblgen", "clang-tblgen")
tblgenRule = pctx.StaticRule("tblgenRule", blueprint.RuleParams{
Depfile: "${out}.d",
Deps: blueprint.DepsGCC,
Command: "${clangTblgen} ${includes} ${generator} -d ${depfile} -o ${out} ${in}",
CommandDeps: []string{"${clangTblgen}"},
Description: "Clang TableGen $in => $out",
}, "includes", "depfile", "generator")
)
type tblgenProperties struct {
In string
Outs []string
}
type tblgen struct {
android.ModuleBase
properties tblgenProperties
exportedHeaderDirs android.Paths
generatedHeaders android.Paths
}
var _ genrule.SourceFileGenerator = (*tblgen)(nil)
func (t *tblgen) GenerateAndroidBuildActions(ctx android.ModuleContext) {
in := android.PathForModuleSrc(ctx, t.properties.In)
includes := []string{
"-I " + ctx.ModuleDir(),
"-I external/llvm/include",
"-I external/llvm/lib/Target",
"-I external/clang/include",
"-I " + filepath.Dir(in.String()),
}
for _, o := range t.properties.Outs {
out := android.PathForModuleGen(ctx, o)
generator := outToGenerator(ctx, o)
ctx.ModuleBuild(pctx, android.ModuleBuildParams{
Rule: tblgenRule,
Input: in,
Output: out,
Args: map[string]string{
"includes": strings.Join(includes, " "),
"generator": generator,
},
})
t.generatedHeaders = append(t.generatedHeaders, out)
}
t.exportedHeaderDirs = append(t.exportedHeaderDirs, android.PathForModuleGen(ctx, ""))
}
func outToGenerator(ctx android.ModuleContext, out string) string {
out = filepath.Base(out)
switch {
case out == "AttrDump.inc":
return "-gen-clang-attr-dump"
case out == "AttrImpl.inc":
return "-gen-clang-attr-impl"
case out == "AttrHasAttributeImpl.inc":
return "-gen-clang-attr-has-attribute-impl"
case out == "AttrList.inc":
return "-gen-clang-attr-list"
case out == "AttrSpellingListIndex.inc":
return "-gen-clang-attr-spelling-index"
case out == "AttrPCHRead.inc":
return "-gen-clang-attr-pch-read"
case out == "AttrPCHWrite.inc":
return "-gen-clang-attr-pch-write"
case out == "Attrs.inc":
return "-gen-clang-attr-classes"
case out == "AttrParserStringSwitches.inc":
return "-gen-clang-attr-parser-string-switches"
case out == "AttrVisitor.inc":
return "-gen-clang-attr-ast-visitor"
case out == "AttrParsedAttrKinds.inc":
return "-gen-clang-attr-parsed-attr-kinds"
case out == "AttrParsedAttrImpl.inc":
return "-gen-clang-attr-parsed-attr-impl"
case out == "AttrParsedAttrList.inc":
return "-gen-clang-attr-parsed-attr-list"
case out == "AttrTemplateInstantiate.inc":
return "-gen-clang-attr-template-instantiate"
case out == "Checkers.inc":
return "-gen-clang-sa-checkers"
case out == "CommentCommandInfo.inc":
return "-gen-clang-comment-command-info"
case out == "CommentCommandList.inc":
return "-gen-clang-comment-command-list"
case out == "CommentHTMLNamedCharacterReferences.inc":
return "-gen-clang-comment-html-named-character-references"
case out == "CommentHTMLTagsProperties.inc":
return "-gen-clang-comment-html-tags-properties"
case out == "CommentHTMLTags.inc":
return "-gen-clang-comment-html-tags"
case out == "CommentNodes.inc":
return "-gen-clang-comment-nodes"
case strings.HasPrefix(out, "Diagnostic") && strings.HasSuffix(out, "Kinds.inc"):
component := strings.TrimPrefix(strings.TrimSuffix(out, "Kinds.inc"), "Diagnostic")
return "-gen-clang-diags-defs -clang-component=" + component
case out == "DiagnosticGroups.inc":
return "-gen-clang-diag-groups"
case out == "DiagnosticIndexName.inc":
return "-gen-clang-diag-groups"
case out == "DeclNodes.inc":
return "-gen-clang-decl-nodes"
case out == "StmtNodes.inc":
return "-gen-clang-stmt-nodes"
case out == "arm_neon.inc":
return "-gen-arm-neon-sema"
case out == "arm_neon.h":
return "-gen-arm-neon"
}
ctx.ModuleErrorf("couldn't map output file %q to a generator", out)
return ""
}
func (t *tblgen) DepsMutator(ctx android.BottomUpMutatorContext) {
}
func (t *tblgen) GeneratedHeaderDirs() android.Paths {
return t.exportedHeaderDirs
}
func (t *tblgen) GeneratedSourceFiles() android.Paths {
return t.generatedHeaders
}
func clangTblgenFactory() (blueprint.Module, []interface{}) {
t := &tblgen{}
return android.InitAndroidModule(t, &t.properties)
}