iar.py 6.1 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
#
# File      : iar.py
# This file is part of RT-Thread RTOS
# COPYRIGHT (C) 2006 - 2015, RT-Thread Development Team
#
#  This program is free software; you can redistribute it and/or modify
#  it under the terms of the GNU General Public License as published by
#  the Free Software Foundation; either version 2 of the License, or
#  (at your option) any later version.
#
#  This program is distributed in the hope that it will be useful,
#  but WITHOUT ANY WARRANTY; without even the implied warranty of
#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
#  GNU General Public License for more details.
#
#  You should have received a copy of the GNU General Public License along
#  with this program; if not, write to the Free Software Foundation, Inc.,
#  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
#
# Change Logs:
# Date           Author       Notes
# 2015-01-20     Bernard      Add copyright information
#

25 26 27
import os
import sys
import string
28
import utils
29

30 31
from SCons.Script import *

32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
import xml.etree.ElementTree as etree
from xml.etree.ElementTree import SubElement
from utils import _make_path_relative
from utils import xml_indent

fs_encoding = sys.getfilesystemencoding()

iar_workspace = '''<?xml version="1.0" encoding="iso-8859-1"?>

<workspace>
  <project>
    <path>$WS_DIR$\%s</path>
  </project>
  <batchBuild/>
</workspace>


'''

def IARAddGroup(parent, name, files, project_path):
    group = SubElement(parent, 'group')
    group_name = SubElement(group, 'name')
    group_name.text = name
X
xieyangrun 已提交
55

56 57 58 59 60 61 62
    for f in files:
        fn = f.rfile()
        name = fn.name
        path = os.path.dirname(fn.abspath)
        basename = os.path.basename(path)
        path = _make_path_relative(project_path, path)
        path = os.path.join(path, name)
X
xieyangrun 已提交
63

64 65
        file = SubElement(group, 'file')
        file_name = SubElement(file, 'name')
X
xieyangrun 已提交
66

67
        if os.path.isabs(path):
68
            file_name.text = path # path.decode(fs_encoding)
69
        else:
70
            file_name.text = '$PROJ_DIR$\\' + path # ('$PROJ_DIR$\\' + path).decode(fs_encoding)
71 72

def IARWorkspace(target):
X
xieyangrun 已提交
73
    # make an workspace
74
    workspace = target.replace('.ewp', '.eww')
75
    out = open(workspace, 'w')
76 77 78
    xml = iar_workspace % target
    out.write(xml)
    out.close()
X
xieyangrun 已提交
79

80 81 82 83 84 85
def IARProject(target, script):
    project_path = os.path.dirname(os.path.abspath(target))

    tree = etree.parse('template.ewp')
    root = tree.getroot()

86
    out = open(target, 'w')
87 88 89 90

    CPPPATH = []
    CPPDEFINES = []
    LINKFLAGS = ''
L
liukangcc 已提交
91
    CFLAGS = ''
92
    Libs = []
X
xieyangrun 已提交
93 94 95 96 97 98 99 100 101 102 103 104 105
    lib_prefix = ['lib', '']
    lib_suffix = ['.a', '.o', '']

    def searchLib(group):
        for path_item in group['LIBPATH']:
            for prefix_item in lib_prefix:
                for suffix_item in lib_suffix:
                    lib_full_path = os.path.join(path_item, prefix_item + item + suffix_item)
                    if os.path.isfile(lib_full_path):
                        return lib_full_path
        else:
            return ''

106 107 108 109 110
    # add group
    for group in script:
        IARAddGroup(root, group['name'], group['src'], project_path)

        # get each include path
111
        if 'CPPPATH' in group and group['CPPPATH']:
112
            CPPPATH += group['CPPPATH']
X
xieyangrun 已提交
113

114
        # get each group's definitions
115
        if 'CPPDEFINES' in group and group['CPPDEFINES']:
116
            CPPDEFINES += group['CPPDEFINES']
X
xieyangrun 已提交
117

118
        # get each group's link flags
119
        if 'LINKFLAGS' in group and group['LINKFLAGS']:
120
            LINKFLAGS += group['LINKFLAGS']
X
xieyangrun 已提交
121

122
        if 'LIBS' in group and group['LIBS']:
123
            for item in group['LIBS']:
X
xieyangrun 已提交
124
                lib_path = searchLib(group)
125 126 127
                if lib_path != '':
                    lib_path = _make_path_relative(project_path, lib_path)
                    Libs += [lib_path]
X
xieyangrun 已提交
128 129 130
                    # print('found lib isfile: ' + lib_path)
                else:
                    print('not found LIB: ' + item)
131

X
xieyangrun 已提交
132
    # make relative path
133 134 135 136
    paths = set()
    for path in CPPPATH:
        inc = _make_path_relative(project_path, os.path.normpath(path))
        paths.add(inc) #.replace('\\', '/')
137

138 139 140 141 142
    # setting options
    options = tree.findall('configuration/settings/data/option')
    for option in options:
        # print option.text
        name = option.find('name')
143

144
        if name.text == 'CCIncludePath2' or name.text == 'newCCIncludePaths':
145 146
            for path in paths:
                state = SubElement(option, 'state')
147
                if os.path.isabs(path) or path.startswith('$'):
148 149 150 151
                    state.text = path
                else:
                    state.text = '$PROJ_DIR$\\' + path

152 153 154 155
        if name.text == 'CCDefines':
            for define in CPPDEFINES:
                state = SubElement(option, 'state')
                state.text = define
156 157 158 159 160 161 162 163 164 165

        if name.text == 'IlinkAdditionalLibs':
            for path in Libs:
                state = SubElement(option, 'state')
                if os.path.isabs(path) or path.startswith('$'):
                    path = path.decode(fs_encoding)
                else:
                    path = ('$PROJ_DIR$\\' + path).decode(fs_encoding)
                state.text = path

166
    xml_indent(root)
167
    out.write(etree.tostring(root, encoding='utf-8').decode())
168
    out.close()
169

170
    IARWorkspace(target)
X
xieyangrun 已提交
171

172 173
def IARPath():
    import rtconfig
174

175 176 177 178
    # backup environ
    old_environ = os.environ
    os.environ['RTT_CC'] = 'iar'
    utils.ReloadModule(rtconfig)
179

180 181
    # get iar path
    path = rtconfig.EXEC_PATH
182

183 184 185
    # restore environ
    os.environ = old_environ
    utils.ReloadModule(rtconfig)
186

187
    return path
188

189 190 191
def IARVersion():
    import subprocess
    import re
192

193
    path = IARPath()
194 195 196 197

    if os.path.exists(path):
        cmd = os.path.join(path, 'iccarm.exe')
    else:
198
        print('Error: get IAR version failed. Please update the IAR installation path in rtconfig.py!')
199
        exit(-1)
200 201 202

    child = subprocess.Popen([cmd, '--version'], stdout=subprocess.PIPE, stderr=subprocess.PIPE, shell=True)
    stdout, stderr = child.communicate()
203 204
    if not isinstance(stdout, str):
        stdout = str(stdout, 'utf8') # Patch for Python 3
205
    # example stdout: IAR ANSI C/C++ Compiler V8.20.1.14183/W32 for ARM
206 207
    iar_version = re.search('[\d\.]+', stdout).group(0)
    return iar_version