org.apache.royale.compiler.clients.ExternCConfiguration Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of compiler-externc Show documentation
Show all versions of compiler-externc Show documentation
The Apache Royale Compiler Externs Compiler
The newest version!
/*
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.royale.compiler.clients;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import org.apache.royale.compiler.config.Configuration;
import org.apache.royale.compiler.config.ConfigurationValue;
import org.apache.royale.compiler.exceptions.ConfigurationException.CannotOpen;
import org.apache.royale.compiler.exceptions.ConfigurationException.IncorrectArgumentCount;
import org.apache.royale.compiler.internal.codegen.typedefs.pass.ReferenceCompiler.TypedefFile;
import org.apache.royale.compiler.internal.codegen.typedefs.reference.BaseReference;
import org.apache.royale.compiler.internal.codegen.typedefs.reference.ClassReference;
import org.apache.royale.compiler.internal.codegen.typedefs.reference.FieldReference;
import org.apache.royale.compiler.internal.codegen.typedefs.reference.MemberReference;
import org.apache.royale.compiler.internal.config.annotations.Arguments;
import org.apache.royale.compiler.internal.config.annotations.Config;
import org.apache.royale.compiler.internal.config.annotations.InfiniteArguments;
import org.apache.royale.compiler.internal.config.annotations.Mapping;
import org.apache.royale.utils.FilenameNormalization;
public class ExternCConfiguration extends Configuration
{
private File jsRoot;
private File asRoot;
private File asClassRoot;
private File asInterfaceRoot;
private File asFunctionRoot;
private File asConstantRoot;
private File asTypeDefRoot;
private File asDuplicatesRoot;
private List typedefs = new ArrayList();
private List externalTypedefs = new ArrayList();
private List namedModules = new ArrayList();
private List callableInstances = new ArrayList();
private List classToFunctions = new ArrayList();
private List excludesClass = new ArrayList();
private List excludesField = new ArrayList();
private List excludes = new ArrayList();
private List readonly = new ArrayList();
private List trueConstants = new ArrayList();
public ExternCConfiguration()
{
}
public File getAsRoot()
{
return asRoot;
}
@Config
@Mapping("as-root")
public void setASRoot(ConfigurationValue cfgval, String filename) throws CannotOpen
{
setASRoot(new File(FilenameNormalization.normalize(getOutputPath(cfgval, filename))));
}
public void setASRoot(File file)
{
this.asRoot = file;
asClassRoot = new File(asRoot, "classes");
asInterfaceRoot = new File(asRoot, "interfaces");
asFunctionRoot = new File(asRoot, "functions");
asConstantRoot = new File(asRoot, "constants");
asTypeDefRoot = new File(asRoot, "typedefs");
asDuplicatesRoot = new File(asRoot, "duplicates");
}
public File getAsClassRoot()
{
return asClassRoot;
}
public File getAsInterfaceRoot()
{
return asInterfaceRoot;
}
public File getAsFunctionRoot()
{
return asFunctionRoot;
}
public File getAsConstantRoot()
{
return asConstantRoot;
}
public File getAsTypeDefRoot()
{
return asTypeDefRoot;
}
public File getAsDuplicatesRoot()
{
return asDuplicatesRoot;
}
public Collection getTypedefs()
{
return typedefs;
}
public Collection getTypedefTypedefs()
{
return externalTypedefs;
}
public boolean isClassToFunctions(String className)
{
return classToFunctions.contains(className);
}
public void addClassToFunction(String className)
{
classToFunctions.add(className);
}
public void addTypedef(File file) throws IOException
{
if (!file.exists())
throw new IOException(file.getAbsolutePath() + " does not exist.");
typedefs.add(new TypedefFile(file));
}
public void addTypedef(String externalFile) throws IOException
{
addTypedef(new File(FilenameNormalization.normalize(externalFile)));
}
public void addExternalTypedef(File file) throws IOException
{
if (!file.exists())
throw new IOException(file.getAbsolutePath() + " does not exist.");
externalTypedefs.add(new TypedefFile(file));
}
public void addExternalTypedef(String externalFile) throws IOException
{
addExternalTypedef(new File(FilenameNormalization.normalize(externalFile)));
}
@Config(allowMultiple = true)
@Mapping("class-to-function")
@Arguments(Arguments.CLASS)
public void setClassToFunctions(ConfigurationValue cfgval, List values) throws IncorrectArgumentCount
{
addClassToFunction(values.get(0));
}
@Config(allowMultiple = true, isPath = true)
@Mapping("typedefs")
@Arguments(Arguments.PATH_ELEMENT)
@InfiniteArguments
public void setTypedef(ConfigurationValue cfgval, String[] vals) throws IOException, CannotOpen
{
for (String val : vals)
addTypedef(resolvePathStrict(val, cfgval));
}
@Config(allowMultiple = true, isPath = true)
@Mapping("external-typedefss")
@Arguments(Arguments.PATH_ELEMENT)
@InfiniteArguments
public void setExternalTypedefs(ConfigurationValue cfgval, String[] vals) throws IOException, CannotOpen
{
for (String val : vals)
addExternalTypedef(resolvePathStrict(val, cfgval));
}
public boolean isExternalTypedef(BaseReference reference)
{
String sourceFileName = reference.getNode().getSourceFileName();
for (TypedefFile file : externalTypedefs)
{
if (sourceFileName.equals("[" + file.getName() + "]"))
{
return true;
}
}
return false;
}
public ExcludedMember isExcludedClass(BaseReference classReference)
{
for (ExcludedMember memeber : excludesClass)
{
if (memeber.isExcluded(classReference, null))
return memeber;
}
return null;
}
public ExcludedMember isExcludedMember(ClassReference classReference,
MemberReference memberReference)
{
if (memberReference instanceof FieldReference)
{
for (ExcludedMember memeber : excludesField)
{
if (memeber.isExcluded(classReference, memberReference))
return memeber;
}
}
for (ExcludedMember memeber : excludes)
{
if (memeber.isExcluded(classReference, memberReference))
return memeber;
}
return null;
}
@Config(allowMultiple = true)
@Mapping("exclude")
@Arguments({"class", "name"})
public void setExcludes(ConfigurationValue cfgval, List values) throws IncorrectArgumentCount
{
final int size = values.size();
if (size % 2 != 0)
throw new IncorrectArgumentCount(size + 1, size, cfgval.getVar(), cfgval.getSource(), cfgval.getLine());
for (int nameIndex = 0; nameIndex < size - 1; nameIndex += 2)
{
final String className = values.get(nameIndex);
final String name = values.get(nameIndex + 1);
addExclude(className, name);
}
}
public void addExclude(String className, String name)
{
excludes.add(new ExcludedMember(className, name));
}
public void addExclude(String className, String name, String description)
{
excludes.add(new ExcludedMember(className, name, description));
}
@Config(allowMultiple = true)
@Mapping("field-exclude")
@Arguments({"class", "field"})
public void setFieldExcludes(ConfigurationValue cfgval, List values) throws IncorrectArgumentCount
{
final int size = values.size();
if (size % 2 != 0)
throw new IncorrectArgumentCount(size + 1, size, cfgval.getVar(), cfgval.getSource(), cfgval.getLine());
for (int nameIndex = 0; nameIndex < size - 1; nameIndex += 2)
{
final String className = values.get(nameIndex);
final String fieldName = values.get(nameIndex + 1);
addFieldExclude(className, fieldName);
}
}
public void addFieldExclude(String className, String fieldName)
{
excludesField.add(new ExcludedMember(className, fieldName, ""));
}
@Config(allowMultiple = true)
@Mapping("class-exclude")
@Arguments("class")
public void setClassExcludes(ConfigurationValue cfgval, List values)
{
for (String className : values)
addClassExclude(className);
}
public void addClassExclude(String className)
{
excludesClass.add(new ExcludedMember(className, null, ""));
}
@Config(allowMultiple = true)
@Mapping("named-module")
@Arguments("module")
@InfiniteArguments
public void setNamedModules(ConfigurationValue cfgval, List values)
{
for (String moduleName : values)
{
addNamedModule(moduleName);
}
}
public void addNamedModule(String moduleName)
{
namedModules.add(moduleName);
}
public String isNamedModule(ClassReference classReference)
{
String basePackageName = classReference.getPackageName();
int packageIndex = basePackageName.indexOf(".");
if (packageIndex != -1)
{
basePackageName = basePackageName.substring(0, packageIndex);
}
for (String module : namedModules)
{
//convert to camel case
String camelCaseModule = convertFromDashesToCamelCase(module);
if(basePackageName.length() == 0)
{
if (classReference.getBaseName().equals(camelCaseModule))
{
return module;
}
continue;
}
if(basePackageName.equals(camelCaseModule))
{
return module;
}
}
return null;
}
/**
* Converts the name of a node module with dashes into a version in camel
* case so that it can be a valid identifier.
*/
private String convertFromDashesToCamelCase(String moduleNameWithDashes)
{
String camelCaseModule = moduleNameWithDashes;
int moduleIndex = camelCaseModule.indexOf("-");
while (moduleIndex != -1 && moduleIndex < camelCaseModule.length() - 1)
{
camelCaseModule = camelCaseModule.substring(0, moduleIndex)
+ camelCaseModule.substring(moduleIndex + 1, moduleIndex + 2).toUpperCase()
+ camelCaseModule.substring(moduleIndex + 2);
moduleIndex = camelCaseModule.indexOf("-");
}
return camelCaseModule;
}
public File getJsRoot()
{
return jsRoot;
}
@Config
@Mapping("js-root")
public void setJSRoot(ConfigurationValue cfgval, String filename) throws CannotOpen
{
this.jsRoot = new File(filename);
}
@Config(allowMultiple = true)
@Mapping("field-readonly")
@Arguments({"class", "name"})
public void setFieldReadOnly(ConfigurationValue cfgval, List values) throws IncorrectArgumentCount
{
final int size = values.size();
if (size % 2 != 0)
throw new IncorrectArgumentCount(size + 1, size, cfgval.getVar(), cfgval.getSource(), cfgval.getLine());
for (int nameIndex = 0; nameIndex < size - 1; nameIndex += 2)
{
final String className = values.get(nameIndex);
final String name = values.get(nameIndex + 1);
readonly.add(new ReadOnlyMember(className, name));
}
}
public ReadOnlyMember isReadOnlyMember(ClassReference classReference,
MemberReference memberReference)
{
for (ReadOnlyMember member : readonly)
{
if ( member.isReadOnly(classReference, memberReference) )
return member;
}
return null;
}
@Config(allowMultiple = true)
@Mapping("true-constant")
@Arguments({"class", "name", "value"})
public void setTrueConstant(ConfigurationValue cfgval, List values) throws IncorrectArgumentCount
{
final int size = values.size();
if (size % 3 != 0)
throw new IncorrectArgumentCount(size + 1, size, cfgval.getVar(), cfgval.getSource(), cfgval.getLine());
for (int nameIndex = 0; nameIndex < size - 2; nameIndex += 3)
{
final String className = values.get(nameIndex);
final String fieldName = values.get(nameIndex + 1);
final String value = values.get(nameIndex + 2);
addTrueConstant(className, fieldName, value);
}
}
public void addTrueConstant(String className, String fieldName, String value)
{
trueConstants.add(new TrueConstant(className, fieldName, value));
}
public TrueConstant isTrueConstant(ClassReference classReference,
MemberReference memberReference)
{
for (TrueConstant constant : trueConstants)
{
if ( constant.isTrueConstant(classReference, memberReference) )
return constant;
}
return null;
}
@Config(allowMultiple = true)
@Mapping("class-callable-instances")
@Arguments("class")
@InfiniteArguments
public void setClassCallableInstances(ConfigurationValue cfgval, List values) throws IOException, CannotOpen
{
callableInstances.addAll(values);
}
public boolean classHasCallableInstances(BaseReference reference)
{
return callableInstances.contains(reference.getQualifiedName());
}
public static class ExcludedMember
{
private String className;
private String name;
private String description;
public String getClassName()
{
return className;
}
public String getName()
{
return name;
}
public String getDescription()
{
return description;
}
public ExcludedMember(String className, String name)
{
this.className = className;
this.name = name;
}
public ExcludedMember(String className, String name, String description)
{
this.className = className;
this.name = name;
this.description = description;
}
public boolean isExcluded(BaseReference classReference,
MemberReference memberReference)
{
if (memberReference == null)
{
return classReference.getQualifiedName().equals(className);
}
return classReference.getQualifiedName().equals(className)
&& memberReference.getQualifiedName().equals(name);
}
public void print(StringBuilder sb)
{
if (description != null)
sb.append("// " + description + "\n");
sb.append("//");
}
}
public static class ReadOnlyMember
{
private String className;
private String name;
public String getClassName()
{
return className;
}
public String getName()
{
return name;
}
public ReadOnlyMember(String className, String name)
{
this.className = className;
this.name = name;
}
public boolean isReadOnly(BaseReference classReference,
MemberReference memberReference)
{
return classReference.getQualifiedName().equals(className)
&& memberReference.getQualifiedName().equals(name);
}
}
public static class TrueConstant
{
private String className;
private String name;
private String value;
public String getClassName()
{
return className;
}
public String getName()
{
return name;
}
public String getValue()
{
return value;
}
public TrueConstant(String className, String name, String value)
{
this.className = className;
this.name = name;
this.value = value;
}
public boolean isTrueConstant(BaseReference classReference,
MemberReference memberReference)
{
return classReference.getQualifiedName().equals(className)
&& memberReference.getQualifiedName().equals(name);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy