Journal Articles
Browse in : |
All
> Journals
> Overload
> 40
(5)
All > Topics > Design (236) Any of these categories - All of these categories |
Note: when you create a new publication type, the articles module will automatically use the templates user-display-[publicationtype].xt and user-summary-[publicationtype].xt. If those templates do not exist when you try to preview or display a new article, you'll get this warning :-) Please place your own templates in themes/yourtheme/modules/articles . The templates will get the extension .xt there.
Title: More Include File Rules
Author: Administrator
Date: 26 December 2000 16:46:04 +00:00 or Tue, 26 December 2000 16:46:04 +00:00
Summary:
Body:
Last issue I presented 15 rules-of thumb, which I keep in mind when working with include files. To be honest, I do not think of these as rules when I code, I just keep to them, only documenting them for Overload has forced me to codify them as 15 rules.
I am very keen that people do not follow these rules slavishly. That is my big problem with coding standards, I do not want to see rules people follow without thinking; I want people to understand what they do. Sometimes, it is important to break from the rules. That is why I was very happy that Kevlin's piece appeared in the same issue. Hopefully, between us we have got people thinking about include files.
To complete what I started, I would like to present a few more rules and discuss some common problems and debugging techniques with include files.
Before I get started I must apologise for a potential error in my last article. In rules 2 and 3 I said that only if a class was declared as a reference or pointer could its include be replaced with a forward declaration. While reading Herb Sutter's Exception C++ I came across his remarks on include file dependencies and he correctly points out that as long as you are only mentioning a class name and not actually using it you may replace an include with a forward declaration.
To clarify my point, suppose we have class Widget and class Thingy, if Thingy contains a Widget, or derives from a Widget it must include the full declaration of Widget. However, if Thingy only mentions Widget in a method (e.g. Action(Widget), Action(const Widget&) or Action(Widget*)), or contains a reference or pointer to Widget, then, it does not need to include Widget's declaration.
What I was trying to say was: if Thingy must access a Widget, prefer a pointer or reference to a Widget rather than containing one. While this is a valid enough rule in isolation, in practice whether Thingy uses reference or value semantics for its Widgets is probably a larger question.
Rule 16: Avoid paths in #include
File locations vary platform to platform: is socket.h in include/ or include/sys/ ? As a general rule it is wise to avoid include paths inside the source code. By placing part of the include path in the source code you are tying the physical locations of source files into the logical source code. If you choose to change your directory structure at a later date you will need to change your source file, which impacts reusablability: not only must the re-user have a copy of your file but they must follow your directory structure.
Any time you reference the directory layout of the system you are coupling your code to the physical directory structure, this is especially true when the parent directory is referenced: e.g. code such as #include "../../utils/helper.h" imposes a lot of directory structure on a project.
Include paths specified outside of the code, through configuration setting, command line options or environment variables break the dependency between file location and file content - it is sensible to use them. On occasions it is necessary to specify part of an include path (e.g. socket.h) but this should not be more than two levels and certainly should not reference the parent directory. (Microsoft Visual C++ 4.0 contained a bug which caused it to corrupt make-files which included '.' paths.)
Rule 17: Break rule 16 when using namespaces
Having said all the above I have been thinking about Kevlin's suggestion that namespaces be placed in a sub-directory and I think he is right. Classically, a project would place most of its headers in a few directories: the system headers, the compiler's headers, third party libraries and one application include. Inevitable the application directory becomes crowded. Now that namespaces provide a way of subdividing the application modules I think echoing these modules in subdirectories is a good idea.
When a system is divided into modules, e.g. ProjectX is divided into gui and xml modules, it makes sense to sub-divide the interfaces (header files) into their own spaces, ProjectX/include/gui and ProjectX/include/xml rather than dump them all in one directory such as ProjectX/include.
Once we have done this we have a choice of either
-
adding all the subdirectories to our include path, hence making it long and complicated, or,
-
placing ProjectX/include in the path and prefixing the filename with the module name, e.g. #inlcude <xml/somefile.h>
Using option 1, if we where to add a new module we would need to change the include path and perhaps effect the entire system. Option 2 follows the both the "least modification" and "only add code" rules, because no global change would be needed and only the files which used the new module would be effected.
Rule 18: Know the difference between local and remote includes
The #include is really two directives: #include "foo.h" and #include <bar.h>: local and remote includes. Traditionally, local meant: search for includes in the same directory as the current compilation unit, i.e. "local is here"; remote meant: search for includes from some include-path - but all this is implementation defined (see K&R or Hatton.)
This could cause problems where the current compilation unit (say bar.cpp) included another file remotely. (say <bar.h>) which itself, included a file locally (say,. "foo.h"). Two problems would occur:
-
Where foo.h exists in the directory containing bar.h but not the one containing bar.cpp : developers can see the file, but the compiler cannot! Hours of frustration.
-
Two different foo.h's existed: one in the same directory as bar.cpp and one in the same directory as foo.h; naturally, one works while one does not. Again, hours of frustration!
To complicate things further, not only is this compiler dependent, but all compilers allow multiple ways of setting include paths, e.g. Microsoft allows "additional include directories" as part of the pre-processor, set for the project (stored in the .DSP file) while "include directories" is set in options/tools and is the same for all project on this machine whether you are compiling the Nuclear Reactor control project, or you Lottery Random number generator.
Some things, like stdio.h are part of the compiler environment regardless of project, some things are always part of the project and some things depend: imagine two versions of the STL on your system: a small fast version (perfect for choosing lottery numbers) and a slow but bug-free version (perfect for fault-tolerant systems).
At some point in your system, the logic in the file must meet the physical file system and directory layout (a subject worthy of an article in its own right.). I have always viewed the glue that bound the two together to be the make-file. The make-file knows how files fit together with each other and the standard gubbins.
My view on include files was very clear:
Rule 18: only use "local include" where the file is local to the includer, chances are it will never been seen by anything else,
Rule 19: use <remote include> for everything else, let the make-file ensure the right thing is seen.
In this world, the Microsoft "additional includes" are part of the makefile and make sense, but they use the "local" mechanism.
I cannot give you a solid rule here: in my mind the jury is still out. Given a blank sheet of paper I
-
use "local" includes for files which are in the same directory and only visible within this module,
-
use <remote> includes for files which form part of the interface of this or another module, and are contained in a separate directory
-
use the environment include paths for system include files, e.g. stdio.h if I am using a standard library supplied with the compiler I would list them here
-
use project include paths for application files and third party files including a third party standard library
-
pay special attention to the standard library include paths: I do not want Plauger's string and Rogue Wave's list.
As mentioned above, the search for header files is implementation dependent. For example, looking for #include "fred.h" Microsoft C++ (5.0) searches:
-
Directory containing the source file
-
Directory containing the file that included the source file, and subsequent searches all the way up the include tree These two rules are skipped for #include <fred.h>, rules 3 and 4 apply to both types of include:
-
Any directory specified with /I on the command line
-
Any directory specified with the INCLUDE environment variable: this corresponds to the options directory list in Developer Studio.
While Sun's C++ (4.2) compiler searches:
-
The directory containing the source file if the include is "local" Rule 1 does not apply for <includes> but the following rules are common:
-
Any directory listed with the -I option on the command line
-
Standard directory for C++ header files; on my 4.2 compiler this is: /opt/SUNWspro/SC4.2/include/CC
-
Standard directory for C header files; again, on my system this is: /opt/SUNWspro/SC4.2/include/cc
-
In /usr/include
Obviously very different from Microsoft, however, GNU C++ (actually egcs 1.1.2 in this case) complicates things further with a multitude of extra options, e.g. two search paths are allowed. Generally, GCC seems closer to Sun but the hard coded paths can be varied at compile time so the same version may actually search in different places.
While I always knew differences existed only when researching this article did I realise how different they can be. Microsoft's upward-recursion algorithm seems particularly dangerous.
Microsoft's compiler offers nothing to help debug include problems, Sun provides an -H option which will produce a list of include files as they are included making it possible to follow the logic. If memory serves GCC/egcs lists which files included which file when some compilation errors occur.
One of the more infuriating problems is when you can see something is included but the compiler will not recognise it. For example:
// file: fred.h class Fred { ... } // file: Flintstones.cpp #include <willma.h> #include <fred.h> Fred fred; // error: undeclared identifier
The most common reasons for this are:
-
The pre-processor actually found a different file called fred.h which does not define Fred: e.g. you are using Microsoft pre-compiled headers and the Visual C++ Wizards, you will have multiple copies of stdafx.h around, something was added to one but it is not the one included here
-
Macro-guards mixed up: typical when cut-and-paste programming is used, e.g. you wrote willma.h and knowing that fred.h was going to be very similar you pasted the code from willma.h into a blank file called fred.h and just made the changes. However, if you forgot to change the #if defined WILMA_H at the start of the file fred.h and hence the pre-processor ignores the contents of fred.h.
The quickest way to find out what is happening to your file is to add a message which identifies it at compile time, e.g.
#pragma message("This is fred.h")
These can be sprinkled around liberally, but remember you will want to remove them before check in. The best places to put these are:
-
Inside the file with the problem: before actually including fred.h, e.g. message("About to include fred.h...") or immediately after including fred.h, e.g. message("... fred.h included")
-
Inside the offending file: before the include guard ("This is fred.h") and inside the include guard ("Inside fred.h include guard").
If you place a message inside the include guard remember that you should only see it once for each compilation unit, e.g. each .cpp file, but you will most likely see it multiple times during a complete build because multiple compilation units will probably include the file.
In practice the pragma message is the most powerful tool for debugging header file issues for most problems. If your compiler does not support pragma message, try #error. Or, add an actual syntax error to the file, these are seldom fatal to the compile but will show you what is happening
Variations on the "it is included" theme are:
-
The file has changed: one of your fellow programmers has revised Fred.h to declare class Fred_Impl instead of Fred.
-
Namespaces: these give us a whole new way to hide identifiers.
Failure to actually find a file is perhaps the most common error. However, it usually has but one course: include paths are wrong. Sometimes only legwork will pay here. Check each any every path individually:
-
If your using Microsoft DSP files, the Settings/Additional include paths dialog box is simply too small to be sure you are not seeing a '1' (one) instead of an 'l' (el), or a '0' (zero) instead of an 'O'. Copy the text and paste it into a bigger window, break the line up and check you can dir each directory at the command line.
-
Check your debug and release settings are the same, are you are building the one you think your building? In the heat of battle it is not known to be looking at the Debug settings when you actually have an error in the Release build
-
If your using makefile these can get quite complex with files including files which include files which all add to compiler and linker options. Look at the command line make is issuing and check the paths.
-
If you rely on an INCLUDE environment variable check the shell. You are executing in actually has it set to what you think it is. If you have su'ed to a different user to build it, it may have been reset.
-
Is your compiler looking for headers in the current directory? (i.e. where the makefile file is) or is it looking in the directory containing the directory containing the source file?. Or for that matter, are you looking in the current directory when you should be looking in the source directory?
Sometimes there are other courses to this error:
-
Your system has run out of file handles; unlikely for NT but some people still develop for MSDOS and even UNIX has limits.
-
Access writes deny access to the file: can you cat the file at the command line? If you get an access denied the compiler will too.
One common error is the type redefinition. This happens when you declare something, e.g. class Table, and something you have previously included declares typedef... Table. Hopefully my rules on include file order should help avoid this. However, this is not always possible.
Developers do not often make this mistake in their own code, but third party code, be it OS headers, bought in libraries, or in house libraries usually contain the other copy. (Hopefully, with namespaces this kind of thing will become less frequent.)
If your lucky you can place your cursor on the offending definition, hit F1 and the on-line help will tell you about the OS type of the name - at which point you must give in gracefully and rename yours. Unfortunately, much that is defined in OS headers is not documented, at which point grepping the code may help.
A quicker way to find the first declaration is too cheat. Add a quick declaration before all your includes, something obscure like: typedef short <Offending identifier>, and then recompile. Normally this will give you the same error, but this time on the first declaration. Once you have removed your extra declaration you will know which items conflict and resolve the conflict.
A particularly nasty form of this involves macros with the same name as types or members, e.g. Sun defines a macro called minor in sysmacros.h, once this file has been included somewhere you will have problems compiling code with CORBA exceptions as these define a member called minor.
As I said right at the start of these articles, including a header file is very primitive. If you write:
// file: fred.h class Fred { ... // <eof> // file: flintstones.h #include <fred.h> ... // <eof>
You will at least get an unexpected end of file error, and possibly a lot of other strange syntax errors. It is obvious here but it is easy to miss because the file it shows up in, in this case, flintstones.h, is a long way from fred.h. Potentially, flintstones.h did not include fred.h directly, but included family.h, which included "characters.h".
Again cut and paste programming is a common cause of this error; legwork, and a few pragma messages are the best attack.
One short cut which you can make is to comment out everything (#includes and source code) in the .cpp file and re-introduce the includes one at a time until the error appears. Check the file you just included, if you are lucky it is guilty, if not repeat the process with its include's too.
The includes of the system map out the dependencies of the system, which files depend on which, which subsystem depends on which. This is one third of your physical design - another third is the library dependencies and the final third your directory structure and make-files (which are the most physical part of your system.)
When designing a system I find it useful to map out the dependencies before hand, find out which parts will depend on which, which are the low level parts of the system, which the high, and which are the same level but independent.
If your dependency strategy is clear, your include files will be clear and you should have no problem following rules such as "most inflexible first, most flexible last". If you find this is not clear you have a potential problem.
I hope the fact that Kevlin and myself have managed to write around 10,000 words on the subject of include files convinces you that there is more here than meets the eye. The fact is, include files are the first point at which your logical design meets the physical build environment and therefore determines much of what is to come.
After this article had been written and prepared for publication, we received the following email from Herb Sutter (yes, for those that had not realised, Overload is read by some of the best in the World). Clearly, we wanted to publish and give Allan a chance to respond. So here is a bit more on the subject from Herb and Allan.
Hi Allan,
I enjoyed your "Include Files" article in Overload issue 39 (September 2000), and would like to share some comments.
Re Rule 2: You argue that forward-declaring the standard "string" typedef yourself is impractical, but it's much stronger than that: It's illegal. The only thing you can do yourself in namespace std is declare specializations of standard templates; you can't declare standard names like basic_string.
Re Rule 3: I agree with the rule, but I don't think your rationale is quite right. You write, "I can use forward declarations with pointers and references, but I[sic] not with [objects]," which is correct if you're talking about function definitions but isn't correct if you're just talking about function declarations (which is what you seem to be talking about in the context of the other Rules). For example, most people are surprised that the following code is a complete header and does not require a definition for class X, because you only need the definition when you need to generate code that must know the size of X (e.g., allocate stack space for a local object or a function value parameter) or call a member function of X (e.g., copy it):
// ...usual include guards here and at // the end... void g( X&, double ); void h( X*, double ); class Y { public: X& f( X& ); // doesn't need definition of X X* f( X* ); // ditto X f( X ); // ditto }; inline X& Y::f( X& r ) { g( r, 1.0 ); // use the reference: still don't need // definition of X } inline X* Y::f( X* p ) { h( p, 1.0 ); // use the pointer: // still don't need definition of X } // can't define Y::f(X) without X's // definition, though, because that // would actually ask the compiler to // generate code to call the copy // constructor and know the object's size
See also Item 26 in Exceptional C++
Re Rule 9: I don't understand this rule. When you say, "Think about include guards, especially for libraries," I can't see how this can apply to non-header files, yet Rule 8 just said that every header should have include guards (and that's right). What is Rule 9 intended to apply to?
Incidentally, I strongly disagree with Lakos' external include guards on two grounds:
-
There's no benefit on most compilers. I admit that I haven't done measurements, as Lakos seems to have done back then, but as far as I know today's compilers already have smarts to avoid the build time reread overhead - even MSVC does this optimization (although it requires you to say "#pragma once"), and it's the weakest compiler in many ways.
-
External include guards violate encapsulation because they require many/all callers to know about the internals of the header - in particular, the special #define name used as a guard. They're also fragile - what if you get the name wrong? what if the name changes?
Re Rule 10: No, never! Even if you disagree with me above and go ahead with external guard, your objections really do apply. If you forget (or mistype) the external guards in even one place...
Re Rules 12-14: Agreed. It's also worth noting that header files should always be written such that they do not depend on #include file ordering (e.g., being included before vs. after any other header files). This actually impacts namespaces and it's a reason I argue that using declarations should never appear anywhere in a header file, not even in a namespace (see my article Migrating To Namespaces in Dr Dobbs Journal, 25(10), October 2000).
Herb
First, thanks for taking the time to read my thoughts and pass on your comments. Interestingly, John Merrells has part two of the piece in the works for Overload 40, in this I added a paragraph as a result of reading Exceptional C++ to clarify rules 2-3. I agree completely with your comments on my rule 3 - forward declarations can be used as long as no code is generated; my writing was a little confused.
In rule 2 (concerning forward declarations in separate files) I was trying to use the string type as an example of how forward declarations could get complex. Unfortunately I chose a bad example. In fact I didn't know re-opening std namespace was illegal, I should have done but I didn't. I wonder how many developers actually know this? Now you've brought it to my attention it's obvious. Perhaps more importantly: do any compilers enforce this rule? I suspect few if any. I suggest we shout it loud: "don't add anything to std namespace."
(This is kind of unfortunate, to my mind, one of the strongest advantages of a namespace over a class is that it can be re-opened and items added but many novices first encounter with namespaces will be with std, one or both points may well be lost on them.)
Rule 9: Reading it again I see I should have phrased it as "Think about external include guards, especially within libraries." My objective was to explore the Lakos argument about external include guards. Like yourself I have problems with Lakos' argument, while he presents a convincing case for improving compile times increasingly I agree with you. In the long run I think external guards make the source code more difficult to read and increase the coupling between a header file and the file's clients. Yet the benefits, if any, will on be apparent on very large projects.
I deliberately said "think about" because I didn't want to give a definite rule, I wanted to make people aware of the argument, to this end your comments are an excellent contribution to the case. Rule 10: I never intended to advocate the removal of internal include guards. We agree here: always leave them in even if you have external guards!
I find it very interesting that you agree with my rules 12 to 14 "order includes with most inflexible first, most flexible last." These rules where the genesis of the article, they have become a fixture of my coding style. However, they go directly against Lakos' advice which states that you should order from local to system. (Not only this, Kevlin's article in the same issue of Overload upholds the Lakos rule.) I hope we have started a debate here.
I hope you'll forgive this novice-writer a few slips and enjoy my continuation of the subject in Overload 40. Once again, many thanks for your comments.
Allan
Notes:
More fields may be available via dynamicdata ..