blob: 45c525f5d43bc74c3eaae4d756a50d21fe09bac3 [file] [log] [blame]
/*
* ======== Platform.xs ========
* Platform support for the generic platform
*/
/*
* ======== Platform.getCpuDataSheet ========
* This Platform's implementation xdc.platform.IPlatform.getCpuDataSheet.
*
* Return the xdc.platform.ICpuDataSheet.Instance object that corresponds
* to the device that runs executables loaded on the specified cpuId.
*/
function getCpuDataSheet(cpuId)
{
var Utils = xdc.useModule('xdc.platform.Utils');
return (Utils.getCpuDataSheet(this.CPU));
}
/*
* ======== Platform.getCreateArgs ========
*/
function getCreateArgs()
{
return (this.$private.args);
}
/*
* ======== Platform.getExeContext ========
* This Platform's implementation xdc.platform.IPlatform.getExeContext.
*
* Return the xdc.platform.IPlatform.ExeContext object that will run the
* specified program prog.
*/
function getExeContext(prog)
{
var Utils = xdc.useModule('xdc.platform.Utils');
var args = this.$private.args;
if (args.verbose > 0) {
print(this.$module.$name + "('"
+ this.$private.name + "') selecting ExeContext for program '"
+ prog.name + "' ...");
print(" endian = " + prog.endian);
print(" codeModel = " + prog.codeModel);
print(" cpuId = " + prog.build.cpuId);
}
/* create an execution context using the "stock" implementation */
var ExeContext = xdc.useModule('xdc.platform.ExeContext');
var cpu = ExeContext.create(this.CPU, this.$module.BOARD);
/* Set the initial memory map from the cpu datasheet. Check if the user
* tried to set cpuArgs, and if it's the case print a deprecation
* warning message.
*/
if (prog.build.cpuArgs != null) {
this.$module.$logWarning("The parameter 'cpuArgs' in xdc.bld.Executable " +
"is deprecated and will not be supported in future releases. " +
"Please, use corresponding platform instance parameters to set " +
"the initial register settings.", this.$module, null);
for (var prop in prog.build.cpuArgs) {
if (!(prop in this.$private.args)) {
this.$private.args[prop] = prog.build.cpuArgs[prop];
}
}
}
cpu.memoryMap = Utils.assembleMemoryMap(cpu, this, true);
// check for the overlap in the memory map
var overlap = Utils.checkOverlap(cpu.memoryMap);
if (overlap != null) {
this.$module.$logError("Memory objects " + overlap + " overlap", this,
overlap);
}
/* assign values to codeMemory and dataMemory */
initDefaultMemory(this, cpu.memoryMap);
Utils.checkDefaults(this, cpu.memoryMap);
return (cpu);
}
/*
* ======== Platform.getExecCmd ========
* This Platform's implementation of xdc.platform.IPlatform.getExecCmd.
*/
function getExecCmd(prog)
{
return ("@$(ECHO) " + this.$package.$name + " cannot run " + prog.name);
}
/*
* ======== Platform.getLinkTemplate ========
* This is Platform's implementation of xdc.platform.IPlatform.getLinkTemplate
*/
function getLinkTemplate(prog)
{
/* use the target's linker command template */
/* we compute the target because this same platform is supported by
* many different tool chains; e.g., ti.targets, gnu.targets, ...
*/
var tname = prog.build.target.$name;
var tpkg = tname.substring(0, tname.lastIndexOf('.'));
var templateName = tpkg.replace(/\./g, "/") + "/linkcmd.xdt";
if (xdc.findFile(templateName) != null) {
return (templateName);
}
else if (tname.indexOf("ti.targets.") == 0) {
return ("ti/targets/linkcmd.xdt");
}
else {
throw xdc.$$XDCException(
this.$package.$name + ".LINK_TEMPLATE_ERROR",
"Target package " + tpkg + " does not contain linker command "
+ "template 'linkcmd.xdt'.");
}
}
/*
* ======== Platform.Instance.init ========
* This function is called to initialize a newly created instance of a
* platform. Platform instances are created just prior to running
* program configuration scripts.
*
* Platform instances may also be created in the build domain.
*
* this.$args.args is the object configured in config.bld via
* Build.platformTable. This is an untyped object that may specify
* platform specific parameters.
*
* this.$args.name is the name used to identify this instance (without
* the package name prefix).
*/
function instance$meta$init(name, args)
{
var args = this.$args.args; /* platform instance args */
var name = this.$args.name; /* platform instance name */
/* cache the platform creation args and platform instance name for the
* getExeContext method above.
*/
this.$private.args = args; /* instance args structure */
this.$private.name = name; /* instance name */
if (args != null) {
if ("deviceName" in args) {
if ((typeof args.deviceName) != "string") {
this.$module.$logError(this.$package.$name +
": 'deviceName' must be a string", this, this.deviceName);
}
this.deviceName = args.deviceName;
}
else {
this.$module.$logError("Parameter deviceName must be specified "
+ " for 'xdc.platform.generic' platform.", this,
this.deviceName);
}
if ("catalogName" in args) {
if ((typeof args.catalogName) != "string") {
this.$module.$logError(this.$package.$name +
": catalog name must be a string", this, this.catalogName);
}
this.catalogName = args.catalogName;
/* since we are a generic platform we don't know the catalog
* package until now, we must dynamically import it here;
* "real" platforms would declare this statically in their
* package.xdc file.
*/
xdc.loadPackage(this.catalogName);
if (!(this.deviceName in xdc.om[this.catalogName])) {
this.$module.$logError("Device " + args.deviceName
+ " does not exist in " + this.catalogName,
this, args.deviceName);
}
}
else {
this.$module.$logError("Parameter catalogName must be specified "
+ " for 'xdc.platform.generic' platform.", this,
this.deviceName);
}
if ("clockRate" in args) {
this.clockRate = args.clockRate;
}
else {
this.$module.$logError("Parameter clockRate must be specified "
+ " for 'xdc.platform.generic' platform.", this,
this.clockRate);
}
if ("sectMap" in args) {
this.sectMap = args.sectMap;
}
if ("customMemoryMap" in args) {
this.customMemoryMap = args.customMemoryMap;
}
if ("renameMap" in args) {
this.renameMap = args.renameMap;
}
if ("externalMemoryMap" in args) {
this.externalMemoryMap = args.externalMemoryMap;
}
if ("codeMemory" in args) {
this.codeMemory = args.codeMemory;
}
if ("dataMemory" in args) {
this.dataMemory = args.dataMemory;
}
if ("stackMemory" in args) {
this.stackMemory = args.stackMemory;
}
this.CPU.deviceName = this.deviceName;
this.CPU.catalogName = this.catalogName;
this.CPU.clockRate = this.clockRate;
}
else {
this.$module.$logError("Parameters deviceName, catalogName and " +
"clockRate must be specified for 'xdc.platform.generic' platform.",
this, args);
}
}
/*
* ======== initDefaultMemory ========
* Initialize platform's config parameters that define in which memory
* segments to place code and data sections
*/
function initDefaultMemory(plat, memMap)
{
/* We need to automatically select codeMemory, dataMemory and stackMemory
* if they are not defined. The three flags defined below are used to
* differentiate between user selected objects and selections made in this
* function. If we already selected a memory object, when we later find
* another one that could be used and has more space, we switch to that
* new one. But, if the user selected a memory object, we don't want to
* redefine it.
*/
var userCodeMem = (plat.codeMemory != null);
var userDataMem = (plat.dataMemory != null);
var userStackMem = (plat.stackMemory != null);
/* The references to the select memory objects are needed to be able to
* quickly read their 'len' parameters.
*/
var codeObj = null;
var dataObj = null;
/* look for a memory of the right type */
for (var j = 0; j < memMap.length; j++) {
var mem = memMap[j];
/* If neither 'space' nor 'page' are defined, the memory segment
* can be used for both, code and data.
*/
if (mem.space == undefined && mem.page == undefined) {
if (!userCodeMem && (codeObj == null || codeObj.len < mem.len)) {
codeObj = mem;
plat.codeMemory = codeObj.name;
}
if (!userDataMem && (dataObj == null || dataObj.len < mem.len)) {
dataObj = mem;
plat.dataMemory = dataObj.name;
}
}
else if (mem.space != undefined) {
if (mem.space.indexOf("code") >= 0) {
if (!userCodeMem &&
(codeObj == null || codeObj.len < mem.len)) {
codeObj = mem;
plat.codeMemory = codeObj.name;
}
}
if (mem.space.indexOf("data") >= 0) {
if (!userDataMem &&
(dataObj == null || dataObj.len < mem.len)) {
dataObj = mem;
plat.dataMemory = dataObj.name;
}
}
}
else if (mem.page != undefined) {
if (mem.page == 0) {
if (!userCodeMem &&
(codeObj == null || codeObj.len < mem.len)) {
codeObj = mem;
plat.codeMemory = codeObj.name;
}
}
if (mem.page == 1) {
if (!userDataMem &&
(dataObj == null || dataObj.len < mem.len)) {
dataObj = mem;
plat.dataMemory = dataObj.name;
}
}
}
}
/* if any of two config parameters is null, we cannot continue because the
* name of the segment for some section would be 'null'.
*/
if (plat.dataMemory == null) {
this.$module.$logError(
"dataMemory in the xdc.platform.generic platform is null");
}
if (plat.codeMemory == null) {
this.$module.$logError(
"codeMemory in the xdc.platform.generic platform is null");
}
if (!userStackMem) {
plat.stackMemory = plat.dataMemory;
}
}