# Copyright (c) 2014-present PlatformIO # # 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. from __future__ import absolute_import from glob import glob from os import environ from os.path import join from SCons.Defaults import processDefines from platformio import util from platformio.managers.core import get_core_package_dir def _dump_includes(env): includes = [] for item in env.get("CPPPATH", []): includes.append(env.subst(item)) # installed libs for lb in env.GetLibBuilders(): includes.extend(lb.get_inc_dirs()) # includes from toolchains p = env.PioPlatform() for name in p.get_installed_packages(): if p.get_package_type(name) != "toolchain": continue toolchain_dir = util.glob_escape(p.get_package_dir(name)) toolchain_incglobs = [ join(toolchain_dir, "*", "include*"), join(toolchain_dir, "*", "include", "c++", "*"), join(toolchain_dir, "*", "include", "c++", "*", "*-*-*"), join(toolchain_dir, "lib", "gcc", "*", "*", "include*") ] for g in toolchain_incglobs: includes.extend(glob(g)) unity_dir = get_core_package_dir("tool-unity") if unity_dir: includes.append(unity_dir) return includes def _get_gcc_defines(env): items = [] try: sysenv = environ.copy() sysenv['PATH'] = str(env['ENV']['PATH']) result = util.exec_command( "echo | %s -dM -E -" % env.subst("$CC"), env=sysenv, shell=True) except OSError: return items if result['returncode'] != 0: return items for line in result['out'].split("\n"): tokens = line.strip().split(" ", 2) if not tokens or tokens[0] != "#define": continue if len(tokens) > 2: items.append("%s=%s" % (tokens[1], tokens[2])) else: items.append(tokens[1]) return items def _dump_defines(env): defines = [] # global symbols for item in processDefines(env.get("CPPDEFINES", [])): defines.append(env.subst(item).replace('\\', '')) # special symbol for Atmel AVR MCU if env['PIOPLATFORM'] == "atmelavr": board_mcu = env.get("BOARD_MCU") if not board_mcu and "BOARD" in env: board_mcu = env.BoardConfig().get("build.mcu") if board_mcu: defines.append( str("__AVR_%s__" % board_mcu.upper() .replace("ATMEGA", "ATmega").replace("ATTINY", "ATtiny"))) # built-in GCC marcos if env.GetCompilerType() == "gcc": defines.extend(_get_gcc_defines(env)) return defines def DumpIDEData(env): LINTCCOM = "$CFLAGS $CCFLAGS $CPPFLAGS $_CPPDEFFLAGS" LINTCXXCOM = "$CXXFLAGS $CCFLAGS $CPPFLAGS $_CPPDEFFLAGS" data = { "libsource_dirs": [env.subst(l) for l in env.get("LIBSOURCE_DIRS", [])], "defines": _dump_defines(env), "includes": _dump_includes(env), "cc_flags": env.subst(LINTCCOM), "cxx_flags": env.subst(LINTCXXCOM), "cc_path": util.where_is_program(env.subst("$CC"), env.subst("${ENV['PATH']}")), "cxx_path": util.where_is_program(env.subst("$CXX"), env.subst("${ENV['PATH']}")), "gdb_path": util.where_is_program(env.subst("$GDB"), env.subst("${ENV['PATH']}")), "prog_path": env.subst("$PROG_PATH"), "compiler_type": env.GetCompilerType() } env_ = env.Clone() # https://github.com/platformio/platformio-atom-ide/issues/34 _new_defines = [] for item in processDefines(env_.get("CPPDEFINES", [])): item = item.replace('\\"', '"') if " " in item: _new_defines.append(item.replace(" ", "\\\\ ")) else: _new_defines.append(item) env_.Replace(CPPDEFINES=_new_defines) data.update({ "cc_flags": env_.subst(LINTCCOM), "cxx_flags": env_.subst(LINTCXXCOM) }) return data def exists(_): return True def generate(env): env.AddMethod(DumpIDEData) return env