Changed preference for selecting package definition files, so that it takes the one from our tree in preference to the one in the package itself. This means that we can just delete items from our tree when we're happy to switch to using the one in the package (as this will vary from package to package).
Added sorting of the attributes when outputting the system definition, to match the de facto standard in the system definition we started with.
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" indent="yes"/>
<xsl:param name="Path">os/deviceplatformrelease/foundation_system/system_model</xsl:param>
<!-- $Path is the location of the root system definition XML file. Must not end in /
This is used to compute the absolute paths the 2.0 syntax needs-->
<xsl:template match="/*">
<xsl:message terminate="yes">Cannot process this document</xsl:message>
</xsl:template>
<!-- can only handle 3.0.0 to 2.0.1 transforms
Assumes only packages are using href
-->
<xsl:template match="/SystemDefinition[@schema='3.0.0']">
<!-- process root system definition or package definition-->
<xsl:call-template name="DTD"/> <!-- insert 2.0.01 DTD -->
<SystemDefinition name="{*/@name}" schema="2.0.1">
<xsl:apply-templates select="*|comment()"/>
</SystemDefinition>
</xsl:template>
<xsl:template match="/SystemDefinition[@schema='3.0.0' and systemModel]">
<xsl:call-template name="DTD"/> <!-- insert 2.0.01 DTD -->
<SystemDefinition name="{systemModel/@name}" schema="2.0.1">
<xsl:apply-templates select="*|comment()"/>
</SystemDefinition>
</xsl:template>
<xsl:template match="/SystemDefinition/package[@href]" priority="2">
<xsl:message terminate="yes">Package definition cannot link another package</xsl:message>
</xsl:template>
<xsl:template match="/SystemDefinition/package" priority="1">
<!-- process package definition file-->
<systemModel>
<layer name="anonymous"> <!-- fake layer -->
<block>
<xsl:apply-templates mode="copy" select="@id|@name|@span|@level|@levels"/><!-- valid attribuites for 2.0 -->
<xsl:apply-templates select="*|comment()">
<xsl:with-param name="path" select="concat($Path,'/')"/> <!-- need to keep tack of where the current document is -->
</xsl:apply-templates>
</block>
</layer>
</systemModel>
</xsl:template>
<xsl:template match="/"><xsl:apply-templates select="*"/></xsl:template>
<xsl:template match="@*|comment()"><xsl:copy-of select="."/></xsl:template>
<!-- comments are copied verbatim. Attribtues are copied by default -->
<xsl:template match="systemModel">
<xsl:copy>
<xsl:apply-templates select="*|comment()"> <!-- no attributes -->
<xsl:with-param name="path" select="$Path"/> <!-- need to keep tack of where the current document is -->
</xsl:apply-templates>
</xsl:copy>
</xsl:template>
<xsl:template mode="copy" match="@*">
<xsl:copy-of select="."/>
</xsl:template>
<xsl:template mode="copy" match="@id"> <!-- id in 3.0 is name in 2.0 -->
<xsl:attribute name="name"><xsl:value-of select="."/></xsl:attribute>
</xsl:template>
<xsl:template mode="copy" match="@name"> <!-- name in 3.0 is long-name in 2.0.1 -->
<xsl:if test=".!=../@id"> <!-- don't bother if it will be the same as name -->
<xsl:attribute name="long-name"><xsl:value-of select="."/></xsl:attribute>
</xsl:if>
</xsl:template>
<xsl:template match="layer"><xsl:param name="path"/>
<layer>
<xsl:apply-templates mode="copy" select="@id|@name|@span|@levels"/> <!-- valid attribuites for 2.0 -->
<xsl:apply-templates select="*|comment()">
<xsl:with-param name="path" select="$path"/>
</xsl:apply-templates>
</layer>
</xsl:template>
<xsl:template match="layer/package"><!-- translates to block -->
<xsl:param name="path"/>
<block>
<xsl:apply-templates mode="copy" select="@id|@name|@span|@level|@levels"/><!-- valid attribuites for 2.0 -->
<xsl:choose>
<xsl:when test="@href">
<xsl:variable name="this" select="."/>
<xsl:for-each select="document(@href,.)/SystemDefinition/*">
<xsl:if test="@id!=$this/@id">
<xsl:message terminate="yes">Error: IDs do not match: <xsl:value-of select="@id"/> vs <xsl:value-of select="$this/@id"/></xsl:message>
</xsl:if>
<xsl:if test="@name and @name!=@id and not($this/@name and $this/@name=$this/@id)">
<!-- set long-name only if name is different from the id and not set in child doc -->
<xsl:attribute name="long-name"><xsl:value-of select="@name"/></xsl:attribute>
</xsl:if>
<xsl:for-each select="@span|@levels|@level">
<!-- copy only if not set in child doc -->
<xsl:if test="not(this/@*[name()=name(current())])">
<xsl:copy-of select="."/>
</xsl:if>
</xsl:for-each>
<xsl:apply-templates select="*|comment()">
<xsl:with-param name="path">
<xsl:call-template name="normpath">
<xsl:with-param name="path" select="concat($path,'/',$this/@href)"/>
</xsl:call-template>
</xsl:with-param>
</xsl:apply-templates>
</xsl:for-each>
</xsl:when>
<xsl:otherwise>
<xsl:apply-templates select="*|comment()">
<xsl:with-param name="path" select="$path"/>
</xsl:apply-templates>
</xsl:otherwise>
</xsl:choose>
</block>
</xsl:template>
<xsl:template match="package/package"> <!-- translates to subblock --><xsl:param name="path"/>
<subblock>
<xsl:apply-templates mode="copy" select="@id|@name"/>
<xsl:apply-templates select="*|comment()">
<xsl:with-param name="path" select="$path"/>
</xsl:apply-templates>
</subblock>
</xsl:template>
<xsl:template match="package/package/pacakge"> <!-- cannot nest this deep --><xsl:param name="path"/>
<xsl:message>Excessive nesting of packages: Ignoring <xsl:value-of select="@id"/></xsl:message>
<xsl:apply-templates select="*|comment()">
<xsl:with-param name="path" select="$path"/>
</xsl:apply-templates>
</xsl:template>
<xsl:template match="collection"><xsl:param name="path"/>
<collection>
<xsl:apply-templates mode="copy" select="@id|@name|@level"/>
<xsl:apply-templates select="*|comment()">
<xsl:with-param name="path" select="$path"/>
</xsl:apply-templates>
</collection>
</xsl:template>
<xsl:template match="component"><xsl:param name="path"/>
<component>
<xsl:apply-templates mode="copy" select="@id|@name|@deprecated|@introduced|@filter|@purpose"/>
<xsl:if test="contains(concat(' ',@class,' '),' plugin ')">
<xsl:attribute name="plugin">Y</xsl:attribute>
</xsl:if>
<xsl:call-template name="class">
<xsl:with-param name="remove">plugin</xsl:with-param>
<xsl:with-param name="add">
<xsl:if test="not(*) and comment()">placeholder</xsl:if>
<xsl:if test="@target='desktop'"> PC</xsl:if>
</xsl:with-param>
</xsl:call-template>
<xsl:apply-templates select="*|comment()">
<xsl:with-param name="path" select="$path"/>
</xsl:apply-templates>
</component>
</xsl:template>
<xsl:template match="unit[@base and not(@mrp or @bldFile)]"/>
<xsl:template match="unit"><xsl:param name="path"/>
<unit>
<xsl:apply-templates select="@mrp|@bldFile|@late">
<xsl:with-param name="path" select="$path"/>
</xsl:apply-templates>
<xsl:copy-of select="@filter|@root|@version|@prebuilt|@priority"/>
</unit>
</xsl:template>
<xsl:template match="unit/@late"> <!-- 2.0 uses Y/N, 3.0 uses yes/no -->
<xsl:attribute name="{name()}">
<xsl:choose>
<xsl:when test=".='yes'">Y</xsl:when>
<xsl:when test=".='no'">N</xsl:when>
</xsl:choose>
</xsl:attribute>
</xsl:template>
<xsl:template match="@mrp|@bldFile"><xsl:param name="path"/>
<xsl:attribute name="{name()}">
<xsl:choose>
<xsl:when test="starts-with(.,'/')"> <!-- keep absolute paths verbatim (barring the leading / ) -->
<xsl:value-of select="substring-after(substring(.,2),'/')"/>
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="normpath">
<xsl:with-param name="path">
<xsl:call-template name="before">
<xsl:with-param name="text" select="$path"/>
</xsl:call-template>
<xsl:value-of select="."/>
</xsl:with-param>
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:attribute>
</xsl:template>
<xsl:template match="meta"/> <!-- strip all meta tags -->
<xsl:template match="meta[info/@contract]"> <!-- except contract -->
<xsl:copy-of select="info/@contract"/>
</xsl:template>
<xsl:template name="class"><xsl:param name="remove"/><xsl:param name="add"/>
<!-- returns the value of the class attribute with the space-separated list of names in $remove taken out and those in $add added on (does not check for duplicates) -->
<xsl:param name="class" select="normalize-space(@class)"/>
<xsl:variable name="r">
<xsl:text> </xsl:text>
<xsl:choose>
<xsl:when test="contains($remove,' ')"><xsl:value-of select="substring-before($remove,' ')"/></xsl:when>
<xsl:otherwise><xsl:value-of select="$remove"/></xsl:otherwise>
</xsl:choose>
<xsl:text> </xsl:text>
</xsl:variable>
<xsl:variable name="c">
<xsl:choose>
<xsl:when test="contains(concat(' ',$class,' '),$r)">
<xsl:value-of select="substring-before(concat(' ',$class,' '),$r)"/>
<xsl:text> </xsl:text>
<xsl:value-of select="substring-after(concat(' ',$class,' '),$r)"/>
</xsl:when>
<xsl:otherwise><xsl:value-of select="$class"/></xsl:otherwise>
</xsl:choose>
<xsl:if test="normalize-space($add)!=''"><xsl:value-of select="concat(' ',normalize-space($add))"/></xsl:if>
</xsl:variable>
<xsl:choose>
<xsl:when test="contains($remove,' ')">
<xsl:call-template name="class">
<xsl:with-param name="remove" select="substring-after($remove,' ')"/>
<xsl:with-param name="class" select="$c"/>
</xsl:call-template>
</xsl:when>
<xsl:when test="normalize-space($c)!=''">
<xsl:attribute name="class">
<xsl:value-of select="normalize-space($c)"/>
</xsl:attribute>
</xsl:when>
</xsl:choose>
</xsl:template>
<xsl:template name="normpath"><xsl:param name="path"/>
<!-- normalize out any ".." in the path in $path -->
<xsl:choose>
<xsl:when test="contains($path,'/../')">
<xsl:call-template name="normpath">
<xsl:with-param name="path">
<xsl:call-template name="before">
<xsl:with-param name="text" select="substring-before($path,'/../')"/>
</xsl:call-template>
<xsl:value-of select="substring-after($path,'/../')"/>
</xsl:with-param>
</xsl:call-template>
</xsl:when>
<xsl:otherwise><xsl:value-of select="$path"/></xsl:otherwise>
</xsl:choose>
</xsl:template>
<!-- return all text before the last / -->
<xsl:template name="before"><xsl:param name="text"/>
<xsl:if test="contains($text,'/')">
<xsl:value-of select="substring-before($text,'/')"/>/<xsl:call-template name="before"><xsl:with-param name="text" select="substring-after($text,'/')"/></xsl:call-template>
</xsl:if>
</xsl:template>
<xsl:template name="DTD">
<xsl:text disable-output-escaping="yes"><![CDATA[<!DOCTYPE SystemDefinition [
<!ELEMENT SystemDefinition ( systemModel )>
<!ATTLIST SystemDefinition
name CDATA #REQUIRED
schema CDATA #REQUIRED
>
<!-- all paths are relative to the environment variable specified by the root attribute, or SOURCEROOT if not. -->
<!-- System Model Section of DTD -->
<!ELEMENT systemModel (layer+)>
<!ELEMENT layer (block* | collection*)*>
<!-- Kernel Services, Base Services, OS Services, Etc -->
<!ATTLIST layer
name CDATA #REQUIRED
long-name CDATA #IMPLIED
levels NMTOKENS #IMPLIED
span CDATA #IMPLIED
>
<!ELEMENT block (subblock* | collection*)*>
<!-- Generic OS services, Comms Services, etc -->
<!ATTLIST block
levels NMTOKENS #IMPLIED
span CDATA #IMPLIED
level NMTOKEN #IMPLIED
name CDATA #REQUIRED
long-name CDATA #IMPLIED
>
<!ELEMENT subblock (collection)*>
<!-- Cellular Baseband Services, Networking Services, etc -->
<!ATTLIST subblock
name CDATA #REQUIRED
long-name CDATA #IMPLIED
>
<!ELEMENT collection (component)*>
<!-- Screen Driver, Content Handling, etc -->
<!ATTLIST collection
name CDATA #REQUIRED
long-name CDATA #IMPLIED
level NMTOKEN #IMPLIED
>
<!ELEMENT component (unit)*>
<!-- contains units or is a package or prebuilt -->
<!ATTLIST component
name CDATA #REQUIRED
long-name CDATA #IMPLIED
deprecated CDATA #IMPLIED
introduced CDATA #IMPLIED
contract CDATA #IMPLIED
plugin (Y|N) "N"
filter CDATA #IMPLIED
class NMTOKENS #IMPLIED
supports CDATA #IMPLIED
purpose ( optional | mandatory | development ) "optional"
>
<!ELEMENT unit EMPTY >
<!-- must be buildable (bld.inf) -->
<!-- bldFile may someday be removed in favour of mrp -->
<!ATTLIST unit
mrp CDATA #IMPLIED
filter CDATA #IMPLIED
bldFile CDATA #IMPLIED
root CDATA #IMPLIED
version NMTOKEN #IMPLIED
prebuilt NMTOKEN #IMPLIED
late (Y|N) #IMPLIED
priority CDATA #IMPLIED
>
]>
]]></xsl:text>
</xsl:template>
</xsl:stylesheet>