Add support for extending the suffixes a scanner knows about.
authorstevenknight <stevenknight@fdb21ef1-2011-0410-befe-b5e4ea1792b1>
Sat, 10 Apr 2004 07:35:32 +0000 (07:35 +0000)
committerstevenknight <stevenknight@fdb21ef1-2011-0410-befe-b5e4ea1792b1>
Sat, 10 Apr 2004 07:35:32 +0000 (07:35 +0000)
git-svn-id: http://scons.tigris.org/svn/scons/trunk@952 fdb21ef1-2011-0410-befe-b5e4ea1792b1

18 files changed:
doc/man/scons.1
src/CHANGES.txt
src/RELEASE.txt
src/engine/SCons/Defaults.py
src/engine/SCons/Environment.py
src/engine/SCons/Scanner/C.py
src/engine/SCons/Scanner/CTests.py
src/engine/SCons/Scanner/D.py
src/engine/SCons/Scanner/Fortran.py
src/engine/SCons/Scanner/IDL.py
src/engine/SCons/Scanner/ScannerTests.py
src/engine/SCons/Scanner/__init__.py
src/engine/SCons/Tool/mingw.py
src/engine/SCons/Tool/msvc.py
test/CPPSUFFIXES.py [new file with mode: 0644]
test/DSUFFIXES.py [new file with mode: 0644]
test/FORTRANSUFFIXES.py [new file with mode: 0644]
test/IDLSUFFIXES.py [new file with mode: 0644]

index 357b829cd7300747e1e332f2a7f1287ae6e6f33f..d271d0ec2b0bb7b879a462310ba59cc271c135a8 100644 (file)
@@ -4197,6 +4197,19 @@ include $_CPPINCFLAGS:
 env = Environment(CCCOM="my_compiler $_CPPINCFLAGS -c -o $TARGET $SOURCE")
 .EE
 
+.IP CPPSUFFIXES
+The list of suffixes of files that will be scanned
+for C preprocessor implicit dependencies
+(#include lines).
+The default list is:
+
+.ES
+[".c", ".C", ".cxx", ".cpp", ".c++", ".cc",
+ ".h", ".H", ".hxx", ".hpp", ".hh",
+ ".F", ".fpp", ".FPP",
+ ".S", ".spp", ".SPP"]
+.EE
+
 .IP CVS
 The CVS executable.
 
@@ -4257,6 +4270,15 @@ depending on the specific C++ compiler being used.
 A function that converts a file name into a Dir instance relative to the
 target being built. 
 
+.IP DSUFFIXES
+The list of suffixes of files that will be scanned
+for imported D package files.
+The default list is:
+
+.ES
+['.d']
+.EE
+
 .IP DVIPDF
 The TeX DVI file to PDF file converter.
 
@@ -4403,6 +4425,16 @@ after first running the file through the C preprocessor.
 Any options specified in the $F77FLAGS and $CPPFLAGS construction variables
 are included on this command line.
 
+.IP FORTRANSUFFIXES
+The list of suffixes of files that will be scanned
+for Fortran implicit dependencies
+(INCLUDE lines).
+The default list is:
+
+.ES
+[".f", ".F", ".for", ".FOR"]
+.EE
+
 .IP File
 A function that converts a file name into a File instance relative to the
 target being built. 
@@ -4417,6 +4449,16 @@ when converting PostScript to PDF files.
 .IP GSCOM
 The Ghostscript command line used to convert PostScript to PDF files.
 
+.IP IDLSUFFIXES
+The list of suffixes of files that will be scanned
+for IDL implicit dependencies
+(#include or import lines).
+The default list is:
+
+.ES
+[".idl", ".IDL"]
+.EE
+
 .IP INCPREFIX
 The prefix used to specify an include directory on the C compiler command
 line.
@@ -5128,7 +5170,7 @@ in any way:  if you want an absolute
 path, you must make it absolute yourself.
 
 .IP SCANNERS
-A list of the available implicit dependency scanners. [CScan] by default.
+A list of the available implicit dependency scanners.
 
 .IP SCCS
 The SCCS executable.
@@ -6903,9 +6945,12 @@ An optional list that can be used to
 determine which scanner should be used for
 a given Node.
 In the usual case of scanning for file names,
-this array will be a list of suffixes
+this argument will be a list of suffixes
 for the different file types that this
 Scanner knows how to scan.
+If the argument is a string,
+then it will be expanded 
+into a list by the current environment.
 
 .IP path_function
 A Python function that takes
index a7323ea7c2191b26fbb04e29d9e73121be6dea17..d7922ce81b5fe9e4e4cb6ba987a5fafacd191369 100644 (file)
@@ -67,6 +67,15 @@ RELEASE 0.96 - XXX
   - Enhance the Task.make_ready() method to create a list of the
     out-of-date Nodes for the task for use by the wrapping interface.
 
+  - Allow Scanners to pull the list of suffixes from the construction
+    environment when the "skeys" keyword argument is a string containing
+    a construction variable to be expanded.
+
+  - Support new $CPPSUFFIXES, $DSUFFIXES $FORTRANSUFFIXES, and
+    $IDLSUFFIXES.  construction variables that contain the default list
+    of suffixes to be scanned by a given type of scanner, allowing these
+    suffix lists to be easily added to or overridden.
+
   From Gary Oberbrunner:
 
   - Add a --debug=presub option to print actions prior to substitution.
index 619c988be277a26405be3832028babf185ddf7e1..09a3d6fe0eae708fbaa39dda902e800ecc8205f4 100644 (file)
@@ -27,6 +27,18 @@ RELEASE 0.96 - XXX
 
   Please note the following important changes since release 0.95:
 
+    - The internal Scanner.add_skey() method longer works for the default
+      scanners, which now use construction variables to hold their lists
+      of suffixes.  If you had a custom Tool specification that was
+      reaching into the internals in this way to add a suffix to one of
+      the following scanner, you must now add the suffix to a construction
+      environment through which you plan to call the scanner, as follows:
+
+
+          CScan.add_skey('.x')       => env.Append(CPPSUFFIXES = ['.x'])
+          DScan.add_skey('.x')       => env.Append(DSUFFIXES = ['.x'])
+          FortranScan.add_skey('.x') => env.Append(FORTRANSUFFIXES = ['.x'])
+
     - The behavior of the env.Append() and env.Prepend() methods has
       changed when appending a string value to a UserList, or vice versa.
       They now behave like normal Python addition of a string to
index 4c7b0cc987ae7cb10f06ff306d7102f96689f87f..94ab8d546345eff1c6bc0782b61596895b5d63eb 100644 (file)
@@ -94,12 +94,25 @@ def SharedFlagChecker(source, target, env):
 
 SharedCheck = SCons.Action.Action(SharedFlagChecker, None)
 
-# Scanners and actions for common language(s).
+# Scanners and suffixes for common languages.
 CScan = SCons.Scanner.C.CScan()
+
+CSuffixes = [".c", ".C", ".cxx", ".cpp", ".c++", ".cc",
+             ".h", ".H", ".hxx", ".hpp", ".hh",
+             ".F", ".fpp", ".FPP",
+             ".S", ".spp", ".SPP"]
+
 DScan = SCons.Scanner.D.DScan()
 
+DSuffixes = ['.d']
+
 FortranScan = SCons.Scanner.Fortran.FortranScan()
 
+FortranSuffixes = [".f", ".F", ".for", ".FOR"]
+
+IDLSuffixes = [".idl", ".IDL"]
+
+# Actions for common languages.
 CAction = SCons.Action.Action("$CCCOM")
 DAction = SCons.Action.Action("$DCOM")
 ShCAction = SCons.Action.Action("$SHCCCOM")
@@ -253,6 +266,10 @@ class NullCmdGenerator:
 ConstructionEnvironment = {
     'BUILDERS'   : {},
     'SCANNERS'   : [CScan, FortranScan, DScan],
+    'CPPSUFFIXES': CSuffixes,
+    'DSUFFIXES'  : DSuffixes,
+    'FORTRANSUFFIXES': FortranSuffixes,
+    'IDLSUFFIXES': IDLSuffixes,
     'PDFPREFIX'  : '',
     'PDFSUFFIX'  : '.pdf',
     'PSPREFIX'   : '',
index 57056ad3686c60bc216d195cda6aa9fe5cdb38b9..2144a7f83bef478487df6663414cf04bf96f07bc 100644 (file)
@@ -384,14 +384,28 @@ class Base:
 
     def get_scanner(self, skey):
         """Find the appropriate scanner given a key (usually a file suffix).
-        Does a linear search. Could be sped up by creating a dictionary if
-        this proves too slow.
         """
-        if self._dict['SCANNERS']:
-            for scanner in self._dict['SCANNERS']:
-                if skey in scanner.skeys:
-                    return scanner
-        return None
+        try:
+            sm = self.scanner_map
+        except AttributeError:
+            try:
+                scanners = self._dict['SCANNERS']
+            except KeyError:
+                return None
+            else:
+                self.scanner_map = sm = {}
+                # Reverse the scanner list so that, if multiple scanners
+                # claim they can scan the same suffix, earlier scanners
+                # in the list will overwrite later scanners, so that
+                # the result looks like a "first match" to the user.
+                scanners.reverse()
+                for scanner in scanners:
+                    for k in scanner.get_skeys(self):
+                        sm[k] = scanner
+        try:
+            return sm[skey]
+        except KeyError:
+            return None
 
     def subst(self, string, raw=0, target=None, source=None, dict=None, conv=None):
         """Recursively interpolates construction variables from the
index 4fcf327cdfb5bf059431f2e1a63dda8485f8473d..a5c2331ea554e72a022d1db2fba58cfa30bad26f 100644 (file)
@@ -36,10 +36,7 @@ def CScan(fs = SCons.Node.FS.default_fs):
     """Return a prototype Scanner instance for scanning source files
     that use the C pre-processor"""
     cs = SCons.Scanner.ClassicCPP("CScan",
-                                  [".c", ".C", ".cxx", ".cpp", ".c++", ".cc",
-                                   ".h", ".H", ".hxx", ".hpp", ".hh",
-                                   ".F", ".fpp", ".FPP",
-                                   ".S", ".spp", ".SPP"],
+                                  "$CPPSUFFIXES",
                                   "CPPPATH",
                                   '^[ \t]*#[ \t]*(?:include|import)[ \t]*(<|")([^>"]+)(>|")',
                                   fs = fs)
index abb72a4444a9a9fb2048ac0154a1d224afdd550b..f63060a330d2868659a5e364de806ef3d0b0c397 100644 (file)
 
 __revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
 
-import TestCmd
-import SCons.Scanner.C
-import unittest
-import sys
 import os
 import os.path
+import sys
+import TestCmd
+import unittest
+import UserDict
+
 import SCons.Node.FS
 import SCons.Warnings
 
+import SCons.Scanner.C
+
 test = TestCmd.TestCmd(workdir = '')
 
 os.chdir(test.workpath(''))
@@ -167,38 +170,29 @@ test.write("f5b.h", "\n")
 
 # define some helpers:
 
-class DummyEnvironment:
-    def __init__(self, listCppPath):
-        self.path = listCppPath
-        
+class DummyEnvironment(UserDict.UserDict):
+    def __init__(self, **kw):
+        UserDict.UserDict.__init__(self)
+        self.data.update(kw)
+
     def Dictionary(self, *args):
-        if not args:
-            return { 'CPPPATH': self.path }
-        elif len(args) == 1 and args[0] == 'CPPPATH':
-            return self.path
-        else:
-            raise KeyError, "Dummy environment only has CPPPATH attribute."
+        return self.data
+
+    def subst(self, strSubst):
+        if strSubst[0] == '$':
+            return self.data[strSubst[1:]]
+        return strSubst
 
-    def subst(self, arg):
-        return arg
+    def subst_list(self, strSubst):
+        if strSubst[0] == '$':
+            return [self.data[strSubst[1:]]]
+        return [[strSubst]]
 
     def subst_path(self, path):
         if type(path) != type([]):
             path = [path]
         return map(self.subst, path)
 
-    def has_key(self, key):
-        return self.Dictionary().has_key(key)
-
-    def __getitem__(self,key):
-        return self.Dictionary()[key]
-
-    def __setitem__(self,key,value):
-        self.Dictionary()[key] = value
-
-    def __delitem__(self,key):
-        del self.Dictionary()[key]
-
 if os.path.normcase('foo') == os.path.normcase('FOO'):
     my_normpath = os.path.normcase
 else:
@@ -217,7 +211,7 @@ def make_node(filename, fs=SCons.Node.FS.default_fs):
 
 class CScannerTestCase1(unittest.TestCase):
     def runTest(self):
-        env = DummyEnvironment([])
+        env = DummyEnvironment(CPPPATH=[])
         s = SCons.Scanner.C.CScan()
         path = s.path(env)
         deps = s(make_node('f1.cpp'), env, path)
@@ -226,7 +220,7 @@ class CScannerTestCase1(unittest.TestCase):
 
 class CScannerTestCase2(unittest.TestCase):
     def runTest(self):
-        env = DummyEnvironment([test.workpath("d1")])
+        env = DummyEnvironment(CPPPATH=[test.workpath("d1")])
         s = SCons.Scanner.C.CScan()
         path = s.path(env)
         deps = s(make_node('f1.cpp'), env, path)
@@ -235,7 +229,7 @@ class CScannerTestCase2(unittest.TestCase):
 
 class CScannerTestCase3(unittest.TestCase):
     def runTest(self):
-        env = DummyEnvironment([test.workpath("d1")])
+        env = DummyEnvironment(CPPPATH=[test.workpath("d1")])
         s = SCons.Scanner.C.CScan()
         path = s.path(env)
         deps = s(make_node('f2.cpp'), env, path)
@@ -244,7 +238,7 @@ class CScannerTestCase3(unittest.TestCase):
 
 class CScannerTestCase4(unittest.TestCase):
     def runTest(self):
-        env = DummyEnvironment([test.workpath("d1"), test.workpath("d1/d2")])
+        env = DummyEnvironment(CPPPATH=[test.workpath("d1"), test.workpath("d1/d2")])
         s = SCons.Scanner.C.CScan()
         path = s.path(env)
         deps = s(make_node('f2.cpp'), env, path)
@@ -253,7 +247,7 @@ class CScannerTestCase4(unittest.TestCase):
         
 class CScannerTestCase5(unittest.TestCase):
     def runTest(self):
-        env = DummyEnvironment([])
+        env = DummyEnvironment(CPPPATH=[])
         s = SCons.Scanner.C.CScan()
         path = s.path(env)
 
@@ -276,8 +270,8 @@ class CScannerTestCase5(unittest.TestCase):
 
 class CScannerTestCase6(unittest.TestCase):
     def runTest(self):
-        env1 = DummyEnvironment([test.workpath("d1")])
-        env2 = DummyEnvironment([test.workpath("d1/d2")])
+        env1 = DummyEnvironment(CPPPATH=[test.workpath("d1")])
+        env2 = DummyEnvironment(CPPPATH=[test.workpath("d1/d2")])
         s = SCons.Scanner.C.CScan()
         path1 = s.path(env1)
         path2 = s.path(env2)
@@ -291,7 +285,7 @@ class CScannerTestCase6(unittest.TestCase):
 class CScannerTestCase8(unittest.TestCase):
     def runTest(self):
         fs = SCons.Node.FS.FS(test.workpath(''))
-        env = DummyEnvironment(["include"])
+        env = DummyEnvironment(CPPPATH=["include"])
         s = SCons.Scanner.C.CScan(fs = fs)
         path = s.path(env)
         deps1 = s(fs.File('fa.cpp'), env, path)
@@ -317,7 +311,7 @@ class CScannerTestCase9(unittest.TestCase):
         SCons.Warnings._warningOut = to
         test.write('fa.h','\n')
         fs = SCons.Node.FS.FS(test.workpath(''))
-        env = DummyEnvironment([])
+        env = DummyEnvironment(CPPPATH=[])
         s = SCons.Scanner.C.CScan(fs=fs)
         path = s.path(env)
         deps = s(fs.File('fa.cpp'), env, path)
@@ -332,7 +326,7 @@ class CScannerTestCase10(unittest.TestCase):
     def runTest(self):
         fs = SCons.Node.FS.FS(test.workpath(''))
         fs.chdir(fs.Dir('include'))
-        env = DummyEnvironment([])
+        env = DummyEnvironment(CPPPATH=[])
         s = SCons.Scanner.C.CScan(fs=fs)
         path = s.path(env)
         test.write('include/fa.cpp', test.read('fa.cpp'))
@@ -351,7 +345,7 @@ class CScannerTestCase11(unittest.TestCase):
         # This was a bug at one time.
         f1=fs.File('include2/jjj.h')
         f1.builder=1
-        env = DummyEnvironment(['include', 'include2'])
+        env = DummyEnvironment(CPPPATH=['include', 'include2'])
         s = SCons.Scanner.C.CScan(fs=fs)
         path = s.path(env)
         deps = s(fs.File('src/fff.c'), env, path)
@@ -365,7 +359,7 @@ class CScannerTestCase12(unittest.TestCase):
         fs.BuildDir('build1', 'src', 1)
         fs.BuildDir('build2', 'src', 0)
         fs.Repository(test.workpath('repository'))
-        env = DummyEnvironment([])
+        env = DummyEnvironment(CPPPATH=[])
         s = SCons.Scanner.C.CScan(fs = fs)
         path = s.path(env)
         deps1 = s(fs.File('build1/aaa.c'), env, path)
@@ -383,7 +377,7 @@ class CScannerTestCase13(unittest.TestCase):
         class SubstEnvironment(DummyEnvironment):
             def subst(self, arg, test=test):
                 return test.workpath("d1")
-        env = SubstEnvironment(["blah"])
+        env = SubstEnvironment(CPPPATH=["blah"])
         s = SCons.Scanner.C.CScan()
         path = s.path(env)
         deps = s(make_node('f1.cpp'), env, path)
@@ -392,7 +386,7 @@ class CScannerTestCase13(unittest.TestCase):
 
 class CScannerTestCase14(unittest.TestCase):
     def runTest(self):
-        env = DummyEnvironment([])
+        env = DummyEnvironment(CPPPATH=[])
         s = SCons.Scanner.C.CScan()
         path = s.path(env)
         deps = s(make_node('f5.c'), env, path)
@@ -401,14 +395,14 @@ class CScannerTestCase14(unittest.TestCase):
 
 class CScannerTestCase15(unittest.TestCase):
     def runTest(self):
-        env = DummyEnvironment([])
-        s = SCons.Scanner.C.CScan()
         suffixes = [".c", ".C", ".cxx", ".cpp", ".c++", ".cc",
                     ".h", ".H", ".hxx", ".hpp", ".hh",
                     ".F", ".fpp", ".FPP",
                     ".S", ".spp", ".SPP"]
+        env = DummyEnvironment(CPPSUFFIXES = suffixes)
+        s = SCons.Scanner.C.CScan()
         for suffix in suffixes:
-            assert suffix in s.skeys, "%s not in skeys" % suffix
+            assert suffix in s.get_skeys(env), "%s not in skeys" % suffix
 
 
 
index 28f3c29dff543c774ba0bd291f7657389429417e..27ba165f41edf4618d9d03c1c39cd6f4a0877f1a 100644 (file)
@@ -32,12 +32,14 @@ Coded by Andy Friesen
 
 __revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
 
+import string
+
 import SCons.Scanner
 
 def DScan(fs = SCons.Node.FS.default_fs):
     """Return a prototype Scanner instance for scanning D source files"""
     ds = DScanner(name = "DScan",
-                  suffixes = ['.d'],
+                  suffixes = '$DSUFFIXES',
                   path_variable = 'DPATH',
                   regex = 'import\s+([^\;]*)\;',
                   fs = fs)
@@ -46,7 +48,7 @@ def DScan(fs = SCons.Node.FS.default_fs):
 class DScanner(SCons.Scanner.Classic):
     def find_include(self, include, source_dir, path):
         # translate dots (package separators) to slashes
-        inc = include.replace('.', '/')
+        inc = string.replace(include, '.', '/')
 
         i = SCons.Node.FS.find_file(inc + '.d',
                                     (source_dir,) + path,
index 473375c811165396ad4774fb78aff265adb2eeec..c29eec5a6745b8f1261135eb3b3d257b8309aa51 100644 (file)
@@ -42,7 +42,7 @@ def FortranScan(fs = SCons.Node.FS.default_fs):
     """Return a prototype Scanner instance for scanning source files
     for Fortran INCLUDE statements"""
     scanner = SCons.Scanner.Classic("FortranScan",
-                                    [".f", ".F", ".for", ".FOR"],
+                                    "$FORTRANSUFFIXES",
                                     "F77PATH",
                                     "(?i)INCLUDE[ \t]+'([\\w./\\\\]+)'",
                                     fs = fs)
index f8769a3731eb77c1b852b2d7ced939d71eafc30e..8c4f68d597d6bd4dbf3b6fbd47ed6cac9db2099d 100644 (file)
@@ -36,7 +36,7 @@ import SCons.Scanner
 def IDLScan(fs = SCons.Node.FS.default_fs):
     """Return a prototype Scanner instance for scanning IDL source files"""
     cs = SCons.Scanner.ClassicCPP("IDLScan",
-                                  [".idl", ".IDL"],
+                                  "$IDLSUFFIXES",
                                   "CPPPATH",
                                   '^[ \t]*(?:#[ \t]*include|[ \t]*import)[ \t]+(<|")([^>"]+)(>|")',
                                   fs = fs)
index 10515803ab6df4f8b163bb85a6a5d034f1c6d4e7..91a0f8c8a50bd3f10ddc3df81a8936c1ec89a4f6 100644 (file)
@@ -34,7 +34,13 @@ class DummyEnvironment(UserDict.UserDict):
         UserDict.UserDict.__init__(self, dict)
         self.data.update(kw)
     def subst(self, strSubst):
+        if strSubst[0] == '$':
+            return self.data[strSubst[1:]]
         return strSubst
+    def subst_list(self, strSubst):
+        if strSubst[0] == '$':
+            return [self.data[strSubst[1:]]]
+        return [[strSubst]]
     def subst_path(self, path):
         if type(path) != type([]):
             path = [path]
@@ -165,6 +171,24 @@ class ScannerTestCase(unittest.TestCase):
         self.failUnless(s.recursive == 1,
                         "did not set recursive flag to 1")
 
+    def test_get_skeys(self):
+        """Test the Scanner.Base get_skeys() method"""
+        s = SCons.Scanner.Base(function = self.func)
+        sk = s.get_skeys()
+        self.failUnless(sk == [],
+                        "did not initialize to expected []")
+
+        s = SCons.Scanner.Base(function = self.func, skeys = ['.1', '.2'])
+        sk = s.get_skeys()
+        self.failUnless(sk == ['.1', '.2'],
+                        "sk was %s, not ['.1', '.2']")
+
+        s = SCons.Scanner.Base(function = self.func, skeys = '$LIST')
+        env = DummyEnvironment(LIST = ['.3', '.4'])
+        sk = s.get_skeys(env)
+        self.failUnless(sk == ['.3', '.4'],
+                        "sk was %s, not ['.3', '.4']")
+
 class CurrentTestCase(unittest.TestCase):
     def test_class(self):
         """Test the Scanner.Current class"""
index c59480e9ed1be6359dfb08d4faae278fe49d73a9..102849db869f53473a3592b5d5421629b8b62fea 100644 (file)
@@ -185,6 +185,12 @@ class Base:
         """Add a skey to the list of skeys"""
         self.skeys.append(skey)
 
+    def get_skeys(self, env=None):
+        if SCons.Util.is_String(self.skeys):
+            return env.subst_list(self.skeys)[0]
+        return self.skeys
+
+
 class Current(Base):
     """
     A class for scanning files that are source files (have no builder)
index fa46652a1125211a458de22a6fcbae5620c66992..1369a8bb7c5d2093726edd0a32ce8233cf3289e7 100644 (file)
@@ -137,9 +137,7 @@ def generate(env):
     env['RCINCPREFIX'] = '--include-dir '
     env['RCINCSUFFIX'] = ''
     env['RCCOM'] = '$RC $RCINCFLAGS $RCFLAGS -i $SOURCE -o $TARGET'
-    CScan = env.get_scanner('.c')
-    if CScan:
-        CScan.add_skey('.rc')
+    env.Append(CPPSUFFIXES = ['.rc'])
     env['BUILDERS']['RES'] = res_builder
     
     # Some setting from the platform also have to be overridden:
index 19fbf1da5ae163b545e6ad9edf032dc0857ad29a..1caa4dbcb8e184b0192d303cafdc9916ba46cbd5 100644 (file)
@@ -448,9 +448,7 @@ def generate(env):
     env['RC'] = 'rc'
     env['RCFLAGS'] = SCons.Util.CLVar('')
     env['RCCOM'] = '$RC $_CPPDEFFLAGS $_CPPINCFLAGS $RCFLAGS /fo$TARGET $SOURCES'
-    CScan = env.get_scanner('.c')
-    if CScan:
-        CScan.add_skey('.rc')
+    env.Append(CPPSUFFIXES = ['.rc'])
     env['BUILDERS']['RES'] = res_builder
 
     try:
diff --git a/test/CPPSUFFIXES.py b/test/CPPSUFFIXES.py
new file mode 100644 (file)
index 0000000..e64dff2
--- /dev/null
@@ -0,0 +1,68 @@
+#!/usr/bin/env python
+#
+# __COPYRIGHT__
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+"""
+Test that we can add filesuffixes to $CPPSUFFIXES.
+"""
+
+import TestSCons
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+env = Environment(CPPPATH = ['.'])
+env.Append(CPPSUFFIXES = ['.x'])
+env.InstallAs('foo_c', 'foo.c')
+env.InstallAs('foo_x', 'foo.x')
+""")
+
+test.write('foo.c', """\
+#include <foo.h>
+""")
+
+test.write('foo.x', """\
+#include <foo.h>
+""")
+
+test.write('foo.h', "foo.h 1\n")
+
+test.run(arguments='.', stdout=test.wrap_stdout("""\
+Install file: "foo.c" as "foo_c"
+Install file: "foo.x" as "foo_x"
+"""))
+
+test.up_to_date(arguments='.')
+
+test.write('foo.h', "foo.h 2\n")
+
+test.run(arguments='.', stdout=test.wrap_stdout("""\
+Install file: "foo.c" as "foo_c"
+Install file: "foo.x" as "foo_x"
+"""))
+
+test.up_to_date(arguments='.')
+
+test.pass_test()
diff --git a/test/DSUFFIXES.py b/test/DSUFFIXES.py
new file mode 100644 (file)
index 0000000..6f10439
--- /dev/null
@@ -0,0 +1,68 @@
+#!/usr/bin/env python
+#
+# __COPYRIGHT__
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+"""
+Test that we can add filesuffixes to $DSUFFIXES.
+"""
+
+import TestSCons
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+env = Environment(DPATH=['.'])
+env.Append(DSUFFIXES = ['.x'])
+env.InstallAs('foo_d', 'foo.d')
+env.InstallAs('foo_x', 'foo.x')
+""")
+
+test.write('foo.d', """\
+import inc;
+""")
+
+test.write('foo.x', """\
+import inc;
+""")
+
+test.write('inc.d', "inc.d 1\n")
+
+test.run(arguments='.', stdout=test.wrap_stdout("""\
+Install file: "foo.d" as "foo_d"
+Install file: "foo.x" as "foo_x"
+"""))
+
+test.up_to_date(arguments='.')
+
+test.write('inc.d', "inc 2\n")
+
+test.run(arguments='.', stdout=test.wrap_stdout("""\
+Install file: "foo.d" as "foo_d"
+Install file: "foo.x" as "foo_x"
+"""))
+
+test.up_to_date(arguments='.')
+
+test.pass_test()
diff --git a/test/FORTRANSUFFIXES.py b/test/FORTRANSUFFIXES.py
new file mode 100644 (file)
index 0000000..1ab7306
--- /dev/null
@@ -0,0 +1,68 @@
+#!/usr/bin/env python
+#
+# __COPYRIGHT__
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+"""
+Test that we can add filesuffixes to $FORTRANSUFFIXES.
+"""
+
+import TestSCons
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+env = Environment()
+env.Append(FORTRANSUFFIXES = ['.x'])
+env.InstallAs('foo_f', 'foo.f')
+env.InstallAs('foo_x', 'foo.x')
+""")
+
+test.write('foo.f', """\
+INCLUDE 'foo.h'
+""")
+
+test.write('foo.x', """\
+INCLUDE 'foo.h'
+""")
+
+test.write('foo.h', "foo.h 1\n")
+
+test.run(arguments='.', stdout=test.wrap_stdout("""\
+Install file: "foo.f" as "foo_f"
+Install file: "foo.x" as "foo_x"
+"""))
+
+test.up_to_date(arguments='.')
+
+test.write('foo.h', "foo.h 2\n")
+
+test.run(arguments='.', stdout=test.wrap_stdout("""\
+Install file: "foo.f" as "foo_f"
+Install file: "foo.x" as "foo_x"
+"""))
+
+test.up_to_date(arguments='.')
+
+test.pass_test()
diff --git a/test/IDLSUFFIXES.py b/test/IDLSUFFIXES.py
new file mode 100644 (file)
index 0000000..2aaf9fe
--- /dev/null
@@ -0,0 +1,70 @@
+#!/usr/bin/env python
+#
+# __COPYRIGHT__
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+"""
+Test that we can add filesuffixes to $IDLSUFFIXES.
+"""
+
+import TestSCons
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+import SCons.Scanner.IDL
+env = Environment(CPPPATH=['.'])
+env.Append(SCANNERS = [ SCons.Scanner.IDL.IDLScan() ],
+           IDLSUFFIXES = ['.x'])
+env.InstallAs('foo_idl', 'foo.idl')
+env.InstallAs('foo_x', 'foo.x')
+""")
+
+test.write('foo.idl', """\
+import <foo.h>
+""")
+
+test.write('foo.x', """\
+#include <foo.h>
+""")
+
+test.write('foo.h', "foo.h 1\n")
+
+test.run(arguments='.', stdout=test.wrap_stdout("""\
+Install file: "foo.idl" as "foo_idl"
+Install file: "foo.x" as "foo_x"
+"""))
+
+test.up_to_date(arguments='.')
+
+test.write('foo.h', "foo.h 2\n")
+
+test.run(arguments='.', stdout=test.wrap_stdout("""\
+Install file: "foo.idl" as "foo_idl"
+Install file: "foo.x" as "foo_x"
+"""))
+
+test.up_to_date(arguments='.')
+
+test.pass_test()