/*
* #%L
* Native ARchive plugin for Maven
* %%
* Copyright (C) 2002 - 2014 NAR Maven Plugin developers.
* %%
* 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.
* #L%
*/
package com.github.maven_nar.cpptasks;
import java.io.File;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Set;
import java.util.Vector;
import org.apache.tools.ant.BuildException;
import org.apache.tools.ant.Project;
import org.apache.tools.ant.types.FlexInteger;
import com.github.maven_nar.cpptasks.compiler.CommandLineLinker;
import com.github.maven_nar.cpptasks.compiler.LinkType;
import com.github.maven_nar.cpptasks.compiler.Linker;
import com.github.maven_nar.cpptasks.compiler.Processor;
import com.github.maven_nar.cpptasks.gcc.GccLinker;
import com.github.maven_nar.cpptasks.types.FlexLong;
import com.github.maven_nar.cpptasks.types.LibrarySet;
import com.github.maven_nar.cpptasks.types.LinkerArgument;
import com.github.maven_nar.cpptasks.types.SystemLibrarySet;
/**
* A linker definition. linker elements may be placed either as children of a
* cc element or the project element. A linker element with an id attribute may
* be referenced by linker elements with refid or extends attributes.
*
* @author Adam Murdoch
* @author Curt Arnold
*/
public class LinkerDef extends ProcessorDef {
private long base;
private String entry;
private Boolean fixed;
private Boolean incremental;
private final Vector librarySets = new Vector();
private Boolean map;
private int stack;
private final Vector sysLibrarySets = new Vector();
private String toolPath;
private String linkerPrefix;
private Boolean skipDepLink;
private final Set<File> libraryDirectories = new LinkedHashSet<>();
/**
* Default constructor
*
* @see java.lang.Object#Object()
*/
public LinkerDef() {
this.base = -1;
this.stack = -1;
}
private void addActiveLibrarySet(final Project project, final Vector libsets, final Vector srcSets) {
final Enumeration srcenum = srcSets.elements();
while (srcenum.hasMoreElements()) {
final LibrarySet set = (LibrarySet) srcenum.nextElement();
if (set.isActive(project)) {
libsets.addElement(set);
}
}
}
private void addActiveSystemLibrarySets(final Project project, final Vector libsets) {
addActiveLibrarySet(project, libsets, this.sysLibrarySets);
}
private void addActiveUserLibrarySets(final Project project, final Vector libsets) {
addActiveLibrarySet(project, libsets, this.librarySets);
}
/**
* Adds a linker command-line arg.
*/
public void addConfiguredLinkerArg(final LinkerArgument arg) {
addConfiguredProcessorArg(arg);
}
/**
* Adds a compiler command-line arg.
*/
public void addConfiguredLinkerParam(final LinkerParam param) {
if (isReference()) {
throw noChildrenAllowed();
}
addConfiguredProcessorParam(param);
}
public boolean addLibraryDirectory(final File directory) {
if (directory == null || !directory.exists()) {
return false;
} else {
return this.libraryDirectories.add(directory);
}
}
public boolean addLibraryDirectory(final File parent, final String path) {
if (parent == null) {
return false;
} else {
final File directory = new File(parent, path);
return addLibraryDirectory(directory);
}
}
public void addLibraryDirectory(final String path) {
final File directory = new File(path);
addLibraryDirectory(directory);
}
/**
* Adds a system library set.
*/
public void addLibset(final LibrarySet libset) {
if (isReference()) {
throw super.noChildrenAllowed();
}
if (libset == null) {
throw new NullPointerException("libset");
}
this.librarySets.addElement(libset);
}
/**
* Adds a system library set.
*/
public void addSyslibset(final SystemLibrarySet libset) {
if (isReference()) {
throw super.noChildrenAllowed();
}
if (libset == null) {
throw new NullPointerException("libset");
}
this.sysLibrarySets.addElement(libset);
}
public void execute() throws org.apache.tools.ant.BuildException {
throw new org.apache.tools.ant.BuildException("Not an actual task, but looks like one for documentation purposes");
}
/**
* Returns an array of active library sets for this linker definition.
*/
public LibrarySet[] getActiveLibrarySets(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef"))
.getActiveUserLibrarySets(defaultProviders, index);
}
final Project p = getProject();
final Vector libsets = new Vector();
for (int i = index; i < defaultProviders.length; i++) {
defaultProviders[i].addActiveUserLibrarySets(p, libsets);
}
addActiveUserLibrarySets(p, libsets);
for (int i = index; i < defaultProviders.length; i++) {
defaultProviders[i].addActiveSystemLibrarySets(p, libsets);
}
addActiveSystemLibrarySets(p, libsets);
final LibrarySet[] sets = new LibrarySet[libsets.size()];
libsets.copyInto(sets);
return sets;
}
/**
* Returns an array of active library sets for this linker definition.
*/
public LibrarySet[] getActiveSystemLibrarySets(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef"))
.getActiveUserLibrarySets(defaultProviders, index);
}
final Project p = getProject();
final Vector libsets = new Vector();
for (int i = index; i < defaultProviders.length; i++) {
defaultProviders[i].addActiveSystemLibrarySets(p, libsets);
}
addActiveSystemLibrarySets(p, libsets);
final LibrarySet[] sets = new LibrarySet[libsets.size()];
libsets.copyInto(sets);
return sets;
}
/**
* Returns an array of active library sets for this linker definition.
*/
public LibrarySet[] getActiveUserLibrarySets(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef"))
.getActiveUserLibrarySets(defaultProviders, index);
}
final Project p = getProject();
final Vector libsets = new Vector();
for (int i = index; i < defaultProviders.length; i++) {
defaultProviders[i].addActiveUserLibrarySets(p, libsets);
}
addActiveUserLibrarySets(p, libsets);
final LibrarySet[] sets = new LibrarySet[libsets.size()];
libsets.copyInto(sets);
return sets;
}
public long getBase(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef")).getBase(defaultProviders, index);
}
if (this.base <= 0 && defaultProviders != null && index < defaultProviders.length) {
return defaultProviders[index].getBase(defaultProviders, index + 1);
}
return this.base;
}
public String getEntry(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef")).getEntry(defaultProviders, index);
}
if (this.entry != null) {
return this.entry;
}
if (defaultProviders != null && index < defaultProviders.length) {
return defaultProviders[index].getEntry(defaultProviders, index + 1);
}
return null;
}
public Boolean getFixed(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef")).getFixed(defaultProviders, index);
}
if (this.fixed == null && defaultProviders != null && index < defaultProviders.length) {
return defaultProviders[index].getFixed(defaultProviders, index + 1);
}
return this.fixed;
}
public boolean getIncremental(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef")).getIncremental(defaultProviders, index);
}
if (this.incremental != null) {
return this.incremental.booleanValue();
}
if (defaultProviders != null && index < defaultProviders.length) {
return defaultProviders[index].getIncremental(defaultProviders, index + 1);
}
return false;
}
public List<File> getLibraryDirectories() {
return new ArrayList<>(this.libraryDirectories);
}
public boolean getMap(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef")).getMap(defaultProviders, index);
}
if (this.map != null) {
return this.map.booleanValue();
}
if (defaultProviders != null && index < defaultProviders.length) {
return defaultProviders[index].getMap(defaultProviders, index + 1);
}
return false;
}
@Override
public Processor getProcessor() {
Linker linker = (Linker) super.getProcessor();
if (linker == null) {
linker = GccLinker.getInstance();
}
if (getLibtool() && linker instanceof CommandLineLinker) {
final CommandLineLinker cmdLineLinker = (CommandLineLinker) linker;
linker = cmdLineLinker.getLibtoolLinker();
}
return linker;
}
@Override
public Processor getProcessor(final LinkType linkType) {
final Processor proc = getProcessor();
return proc.getLinker(linkType);
}
public int getStack(final LinkerDef[] defaultProviders, final int index) {
if (isReference()) {
return ((LinkerDef) getCheckedRef(LinkerDef.class, "LinkerDef")).getStack(defaultProviders, index);
}
if (this.stack < 0 && defaultProviders != null && index < defaultProviders.length) {
return defaultProviders[index].getStack(defaultProviders, index + 1);
}
return this.stack;
}
public String getToolPath() {
return this.toolPath;
}
public String getLinkerPrefix() {
return this.linkerPrefix;
}
public boolean isSkipDepLink() {
return this.skipDepLink.booleanValue();
}
/**
* Sets the base address. May be specified in either decimal or hex.
*
* @param base
* base address
*
*/
public void setBase(final FlexLong base) {
if (isReference()) {
throw tooManyAttributes();
}
this.base = base.longValue();
}
/**
* Sets the starting address.
*
* @param entry
* function name
*/
public void setEntry(final String entry) {
if (isReference()) {
throw tooManyAttributes();
}
this.entry = entry;
}
/**
* If true, marks the file to be loaded only at its preferred address.
*/
public void setFixed(final boolean fixed) {
if (isReference()) {
throw tooManyAttributes();
}
this.fixed = booleanValueOf(fixed);
}
/**
* If true, allows incremental linking.
*
*/
public void setIncremental(final boolean incremental) {
if (isReference()) {
throw tooManyAttributes();
}
this.incremental = booleanValueOf(incremental);
}
/**
* If set to true, a map file will be produced.
*/
public void setMap(final boolean map) {
if (isReference()) {
throw tooManyAttributes();
}
this.map = booleanValueOf(map);
}
/**
* Sets linker type.
*
*
* <table width="100%" border="1">
* <thead>Supported linkers </thead>
* <tr>
* <td>gcc</td>
* <td>Gcc Linker</td>
* </tr>
* <tr>
* <td>g++</td>
* <td>G++ Linker</td>
* </tr>
* <tr>
* <td>ld</td>
* <td>Ld Linker</td>
* </tr>
* <tr>
* <td>ar</td>
* <td>Gcc Librarian</td>
* </tr>
* <tr>
* <td>msvc</td>
* <td>Microsoft Linker</td>
* </tr>
* <tr>
* <td>bcc</td>
* <td>Borland Linker</td>
* </tr>
* <tr>
* <td>df</td>
* <td>Compaq Visual Fortran Linker</td>
* </tr>
* <tr>
* <td>icl</td>
* <td>Intel Linker for Windows (IA-32)</td>
* </tr>
* <tr>
* <td>ecl</td>
* <td>Intel Linker for Windows (IA-64)</td>
* </tr>
* <tr>
* <td>icc</td>
* <td>Intel Linker for Linux (IA-32)</td>
* </tr>
* <tr>
* <td>ecc</td>
* <td>Intel Linker for Linux (IA-64)</td>
* </tr>
* <tr>
* <td>CC</td>
* <td>Sun ONE Linker</td>
* </tr>
* <tr>
* <td>aCC</td>
* <td>HP aC++ Linker</td>
* </tr>
* <tr>
* <td>os390</td>
* <td>OS390 Linker</td>
* </tr>
* <tr>
* <td>os390batch</td>
* <td>OS390 Linker</td>
* </tr>
* <tr>
* <td>os400</td>
* <td>IccLinker</td>
* </tr>
* <tr>
* <td>sunc89</td>
* <td>C89 Linker</td>
* </tr>
* <tr>
* <td>xlC</td>
* <td>VisualAge Linker</td>
* </tr>
* <tr>
* <td>wcl</td>
* <td>OpenWatcom C/C++ linker</td>
* </tr>
* <tr>
* <td>wfl</td>
* <td>OpenWatcom FORTRAN linker</td>
* </tr>
* </table>
*
*/
public void setName(final LinkerEnum name) throws BuildException {
if (isReference()) {
throw tooManyAttributes();
}
final Linker linker = name.getLinker();
super.setProcessor(linker);
}
@Override
protected void setProcessor(final Processor proc) throws BuildException {
Linker linker = null;
if (proc instanceof Linker) {
linker = (Linker) proc;
} else {
final LinkType linkType = new LinkType();
linker = proc.getLinker(linkType);
}
super.setProcessor(linker);
}
/**
* Sets stack size in bytes.
*/
public void setStack(final FlexInteger stack) {
if (isReference()) {
throw tooManyAttributes();
}
this.stack = stack.intValue();
}
public void setToolPath(final String path) {
this.toolPath = path;
}
public void setLinkerPrefix(final String prefix) {
this.linkerPrefix = prefix;
}
public void setSkipDepLink(final boolean skipDepLink) {
this.skipDepLink = booleanValueOf(skipDepLink);
}
public void visitSystemLibraries(final Linker linker, final FileVisitor libraryVisitor) {
final Project p = getProject();
if (p == null) {
throw new java.lang.IllegalStateException("project must be set");
}
if (isReference()) {
final LinkerDef master = (LinkerDef) getCheckedRef(LinkerDef.class, "Linker");
master.visitSystemLibraries(linker, libraryVisitor);
} else {
//
// if this linker extends another,
// visit its libraries first
//
final LinkerDef extendsDef = (LinkerDef) getExtends();
if (extendsDef != null) {
extendsDef.visitSystemLibraries(linker, libraryVisitor);
}
if (this.sysLibrarySets.size() > 0) {
final File[] libpath = linker.getLibraryPath();
for (int i = 0; i < this.sysLibrarySets.size(); i++) {
final LibrarySet set = (LibrarySet) this.sysLibrarySets.elementAt(i);
if (set.isActive(p)) {
set.visitLibraries(p, linker, libpath, libraryVisitor);
}
}
}
}
}
public void visitUserLibraries(final Linker linker, final FileVisitor libraryVisitor) {
final Project p = getProject();
if (p == null) {
throw new java.lang.IllegalStateException("project must be set");
}
if (isReference()) {
final LinkerDef master = (LinkerDef) getCheckedRef(LinkerDef.class, "Linker");
master.visitUserLibraries(linker, libraryVisitor);
} else {
//
// if this linker extends another,
// visit its libraries first
//
final LinkerDef extendsDef = (LinkerDef) getExtends();
if (extendsDef != null) {
extendsDef.visitUserLibraries(linker, libraryVisitor);
}
//
// visit the user libraries
//
if (this.librarySets.size() > 0) {
final File[] libpath = linker.getLibraryPath();
for (int i = 0; i < this.librarySets.size(); i++) {
final LibrarySet set = (LibrarySet) this.librarySets.elementAt(i);
if (set.isActive(p)) {
set.visitLibraries(p, linker, libpath, libraryVisitor);
}
}
}
}
}
}