Previous Next Table of Contents
This section is divided into three parts, Installation, Development Environment, and Running Swarm.
2.1.1. Should I use a binary or archive installation?
Answer: We created one version of Swarm that includes compiled versions of the auxiliary packages tcl, tk, tclobjc, and blt. We require the system to supply the GNU compiler (including Objective-C), Gnu make, the xpm package, and various X-Windows development libraries. It can be installed by properly editing the Makefile.conf. There are different versions of the binary package, so be careful to pick the one that matches the system you have.
Answer: Swarm is supplied with the full source code. If the user has installed tcl, tk, libobjc, and blt in a Unix system, then Swarm can be compiled to use those libraries. This is the "archive" installation method. The libraries are dynamically loaded by the operating system.
Answer: The archive method is more difficult because the user has to configure tcl/tk/blt/libobjtcl, etc. However, the archive user has more control over what options are used in the compilation of these auxiliary libraries. Upgrades to these auxiliary libraries can be made by the user. On some systems, the binary install method may be unsuitable, may conflict with existing software, so the archive method is the only alternative. In addition, patches to the Swarm source code can be applied on archive installs, but not the binary installs.
Answer: You will find the needed packages in the Swarm homepage along with the current release. You will also find it at ftp://ftp.santafe.edu/pub/swarm/required-software/.
Question: Suppose I have a Unix system (ex: Redhat Linux 4.1) that has GCC, GNU make, and xpm installed. How easy is the binary installation of Swarm?
Answer: Try these steps. You tell us how easy they are!
1. Download the swarm distribution (source + binary) and unpack it into a directory, such as /swarm or /home/myaccount/swarm or ..... We refer to that directory after this as $SWARMHOME.
2. Use a text editor to edit Makefile.conf in $SWARMHOME. The line OTHERINCDIRS should give the addresses of header files, while the OTHERINCLIBS should give the addresses of system libraries. I hope to get examples more Makefile.conf for different systems. The Makefile.conf that is provided has some example configurations that work for certain systems. On my Redhat 4.2 linux system, this is the relevant part of the Makefile.conf that worked:
### for SETUP=NONE
SYS=GNU/Linux
OTHERINCDIRS= -I/usr/include/ -I/usr/X11R6/include -I/usr/local/include
OTHERLIBDIRS=-L/usr/lib -L/usr/X11R6/lib -L/usr/local/lib
SYSDEPLIBS= -ldl
LDCONFIG=bin
3. After saving the Makefile.conf, type "make" and a big spew will ensue. If GCC crashes, you will have to fix your Makefile.conf. Otherwise, proceed.
4. Create environment variables. On some machines, this is not necessary, but why don't you make it easy on youself by not finding out if it is by just doing it. (This example is due to Pietro Terna):
The contents of the .bash_profile file are (useful only for the bash
shell):
TK_LIBRARY=/home/your_name/the_dir_where_you_have_swarm/lib
export TK_LIBRARY
TCL_LIBRARY=/home/your_name/the_dir_where_you_have_swarm/lib
export TCL_LIBRARY
In my liux box, this is
TK_LIBRARY=/home/terna/swarm/swarm/lib
export TK_LIBRARY
TCL_LIBRARY=/home/terna/swarm/swarm/lib
export TCL_LIBRARY
This works because, with the binary install, there are copies of the appropriate files included in $SWARMHOME/lib and $SWARMHOME/include.
5. If you didn't do it yet, download swarmapps.tar.gz and unpack into some directory. Currently, we are on version 1.0.2 of swarmapps--get the newest! Try Heatbugs, for example. Edit its Makefile and put in your $SWARMHOME in place of the directory in the top line of the file. Save, type "make." If there are no errors or crashes because Swarm can't find libraries it wants, just type ./heatbugs and click on "GO" to make it simulate. If there are errors, proceed to step 5.
6. Many users report that they get compile errors when they try to make programs. Errors say GCC can't find tcl.init or such. Users have reported solving these problems in novel and interesting ways. If you happened to put Swarm in the exact same directory as the machine on which the binary was built, you should not see those messages. Probably, however, you did not and you need to fix it by specifying environment variables.
Answer: No. Remove that version of blt, get a newer version of BLT from the SFI web site. Versions 2.1 and 2.3 both can be made to work. If you install Swarm 1.0.5 or later, version 2.3 is preferred. For earlier versions of Swarm, there were some problems with version 2.3, but 2.1 worked fine. Version 2.3 is preferred because it has some additional features and it will work without much modification.
Answer: Absolutely not! The Swarm binary version is distributed with tk 4.1 and tcl 7.5, and it should run fine on any system. If you do an archive install, Swarm can be made to work with 4.2 and 7.6. You probably should use BLT 2.3, which will compile and work with the 7.6/4.2 combination just fine. You can use BLT 2.1, as well, but there is one fix that must be made to the source code (see next note).
Answer: In the blt source, find file bltBgexec.c. Line 1178 has to be changed from
numPids = Tcl_CreatePipeline(interp, argc - i, argv + i, &(bgPtr->pidArr),
to
numPids = TclCreateCommandChannel(interp, argc - i, argv + i, &(bgPtr->pidArr),
You can do this by hand with a text editor or you can apply the patch supplied by Glen Ropella, that you can find in the mailing list archive. Be careful- if you use a Microsoft based computer to read and save this patch, it will have the added carriage returns and it will not work when you copy it into the Unix system. You either must view the patch with a native Linux browser or edit out the carriage returns with a program such as NEdit. After you have the patch, put it in your blt2.1/src directory, and type "patch < blt-patch-for-Swarm-with-tcl7.6."
Answer: Possibly. Those RPMs are user-contributed, so there is no guarantee they will match your system. You may need to get the SRPM file (source version) and use the --rebuild option to compile libraries that match your system. The blt2.1 rpm is it is built on top of the incremental tcl/tk packages that will also have to be downloaded. Some users have reported inexplicable problems building Swarm in that environment. The RPM for blt2.3 may work fine, since that source requires no special patches to work with tcl7.6 and tk4.2.
Answer: Yes, there were some compatability issues early-on, but in swarm-1.0.5, those seem to be worked out. blt2.3 seems to work fine with tcl7.6 and tk4.2. No patching required.
Answer: On my system, Redhat 5.0, tcl/tk 8.0 is woven into the fabric of the system, and there's no feasible way to yank it out to install the older libraries. It is possible to install tcl7.6 and tk4.2 in a separate directory structure and then CAREFULLY configure the blt, tclobjc, and swarm software to use those files. There is a step by step guide mailing-list guide for that.
Recently (1-24-98), it has come to my attention that there is a new version of blt2.1 that will work with tcl/tk 8.0. The package is called blt8.0unofficial. The version of Swarm that uses blt8.0unofficial is still in the development stage, but by the time you come across this FAQ, it may be that SFI staff have announced its availability for testing/usage. In a nutshell, here is what will be necessary. First, upgrade to tcl/tk version 8.0p2 (patched version 2), then compile/install blt8.0 (it has good instructions included with it), get a new version of tclobjc and swarm from SFI, compile them, and run the test applications that SFI will have available.
Question: I downloaded Swarm source, the blt2.1 and tclobjc packages, but I can't understand their directions. Do you have an example script to clarify installation?
Answer: To give you an idea of how the installation proceeds on a system using the Redhat 4.1 distribution of Linux (which already includes GCC, Obj-C, make, tcl 7.6, tk 4.2, xpm, and X11 development packages), here are the steps that worked for several users. (Special thanks to Sven Thommesen.) A similar script was written for Redhat 5.0 and the special problems of configuring Swarm to run on a system with 2 versions of tcl/tk. Check the mailing-list archive. In the following, the symbol # means "type this at the command prompt."
1. Build libtclobjc: unpack the source into /usr/src/libtclobjc-1.1b6 (Update: now this is simply called tclobjc-1.2.tar.gz)
# ./configure --prefix=/usr/local
(You will be asked for the locations of tcl and tk, which are both in /usr/lib)
# make
# make install
2. Build the blt source: (I'm assuming you have unpacked the blt source into /usr/src/blt2.1 and applied the BLT patch described above.)
# cd /usr/src/blt2.1
# ./configure --prefix=/usr/local
# make
# make install
(If there's problems, check in config.BLT that: TCLINCDIR=/usr/include, TCLLIBDIR=/usr/include/../lib, TKINCDIR=/usr/include, TKLIBDIR=/usr/include/../lib, XLIBDIR=/usr/X11R6/lib, XINCDIR=/usr/X11R6/include, PREFIX=/usr/local)
3. Make a missing link so the dynamic linker will find BLT (may not be necessary on some systems):
# cd /usr/local/lib
# ln -s libBLT.so.2.1 libBLT.so.2
# ln -s libBLT.so.2.1 libBLT.so
(NOTE: This is necessary because executables look for libBLT.so in the linker path when they run.)
4. Update the linker database. The crudest (and dangerous) method is to edit the file /etc/ld.so.conf and add "/usr/local/lib" as the first line. Then:
# ldconfig
To test the result, do this:
# ldconfig -p
This will print out the database of libraries found. Check that the list includes: libtk.so=> libtk4.2.so, libtcl.so =>libtcl7.6.so, libBLT.so.2 => /usr/local/lib/libBLT.so.2.1
(Updated material here: 2-9-1998) It is SAFER to put /usr/local/lib in your dynamic linker's path by either putting an -rpath statement in the Makefile.conf or by creating an LD_LIBRARY_PATH environment variable. Leave ld.so.conf alone if possible!
In Makefile.conf (see next step), you can add, after SYSDEPLIBS:
override EXTRALDFLAGS+=-Wl,-rpath,/usr/local/lib
This causes the linker to look in /usr/local/lib whenever a Swarm program gets compiled. You can add paths in the same command with additional -rpath commands.
If you don't add that in the Makefile.conf, you can (in a bash shell), type
#export LD_LIBRARY_PATH=/usr/local/lib:$LD_LIBRARY_PATH
This can be put in a profile if you want it to be permanent, of course.
5. Unpack the Swarm distribution into a directory such as /swarm. That directory is hereafter called $SwarmHome. Edit $SwarmHome/Makefile.conf: (this is an example setup, amend as needed! The INCDIRS are .h files, the LIBDIRS are files whare you find libX11.so, libBLT.so, and so forth.)
SETUP=NONE
SYS=GNU/Linux
LDCONFIG=archive
OTHERINCDIRS= -I/usr/X11R6/include/X11 -I/usr/include -I/usr/local/include
OTHERLIBDIRS= -L/usr/X11R6/lib -L/usr/local/lib -L/usr/lib
SYSDEPLIBS= -ldl //This is VITAL for required dynamic linking in Linux!
Type "make" and there should be a long sequence of GCC compile results. If error messages say some file.h cannot be found, you need to fix OTHERINCDIRS. If errors say some file.a or file.so cannot be found, then you need to fix OTHERLIBDIRS.
6. Go into a Swarm application directory, such as heatbugs. (If you don't have them, go get swarmapps.tar.gz on the Web site!) Edit first line of the Makefile to point to the directory where you installed Swarm. Type 'make.' There should be many GCC messages, don't worry. Look for errors like "cannot find libXfoo" or such. Then see what dynamic libraries were linked in by running
# ldd ./app-name
If all went well, the list should be headed by: libBLT.so.2 libtk.so libtcl.so. If not, look below.
Answer: Lamentably, step 5, "Installing BLT," is missing the commands #make and #make install.
Answers: 1. This may mean your setup is missing some development libraries for which BLT is looking. One user reported that this problem disappeared after installing the libraries libX11.so and libX11.a and the header files that go with them. BLT generally finds them when they are installed in standard places. These libraries are in the Redhat distribution in a rpm called XFree86-devel-3.2-4.
2. Other users observe there is a typing mistake in the config.BLT configure script. Instead of telling you what it really wants, X11/Xlib.h, it says it wants X11/X11.h. Even though it says it wants X11.h, which you can't find, just tell it where you find the directory X11 that has Xlib.h. In Redhat Linux, for example, one finds Xlib.h in the directory "/usr/X11R6/include/X11", so the configuration should be told X11/Xlib.h is in "/usr/X11R6/include". The ./configure statement (from a GNU package) has the configuration option --x-includes and --x-libraries that can be used to specify these.
Answer: BLT has to be informed that you have a "686" system, which didn't exist when BLT was created. In the blt source directory, go to subdirectory /cf and edit a file config.sub. Change references [345]86 to [3456]86. Now BLT should make fine.
Answer: No.
collect2: ld returned 4 exit status
ld: -ltk4.0: expected libx.so(.major.minor.xxx)
*** Error code 1
make: Fatal error: Command failed for target `xyz'
Answer: The problem is that some linkers don't work with dotted version numbers in the library name. The quick fix is to link libtk.a to libtk4.0, then link with "-ltk". Ensure that no other libtk.a (especially a 3.6 libtk) is in your load path first.
Answer: The necessary parts are: barchart, bitmap, busy, drag&drop, graph, and BLT_ZoomStack.
Answer: Could be anything. However, we have seen in the past that forgetting to run make install or forgetting to delete the static library (the .a file) after installing on an ELF system can cause this type of behaviour.
Answer: Try adding "-lobjc -ldl" in the Makefile. That solved the make check problems for my Redhat 5.0 installation. If you look at the Makefile, and there are empty spaces for the tcl and tk include & library directories, go ahead and put in the usual -I/usr... and -L/usr/... stuff. This is important if you have more than one version of tcl/tk in your system and it may find one at compile time and a different one at link time unless you specify these.
When running Redhat 4.2, with tcl 7.6 and tk 4.2, some of us never did get the make check to work, but Swarm worked anyway. "Note: I was able to run 'make check', after a fashion: firstly, start X first. Next, under doCheckTcl in the Makefile remove the '<' character. If you say 'make doCheckTk' you get a little button you can click. If you say 'make doCheckTcl' you get the Tcl interpreter, and you can type commands to it. I don't believe the check.tcl script gets run, though. If I just say 'make check', then process doCheckTcl used 99%CPU for over 30 min on my PP200 before I killed it ... But at least we no longer get those reams of error messages! --Sven"
Answer: No. But at least one user has configured it and Swarm
runs fine in a Redhat 4.1 setup. Sven T. says "To get libtclobjc-1.1b6
to include support for the 'readline' library, you have to set environment
variables when configuring libtclobjc:
# make distclean
# READLINE_INCLUDE_DIR="/usr/include/readline" // RedHat 4.1
# export READLINE_INCLUDE_DIR # READLINE_LIBRARY_DIR="/usr/lib"
# export READLINE_LIBRARY_DIR
# ./configure --prefix=/usr/local --with-readline
# make
# make install.
Then, to get Swarm apps to link properly, add "-lreadline -ltermcap"
to the line SYSDEPLIBS in Makefile.conf in $Swarmhome. I did not see any
difference in the way my Swarm app ran, however."
Answer: You are using the old tclobjc software, which was called libtclobjc. You need to get tclobjc-1.2 or newer.
"Tk.h,"
even though we found "tk.h." Where is it?
Answer: You most likely do not have libtclobjc installed such that it's in the compiler's library search path. If you don't have tclobjc at all, install it. If you have it already, then ensure that the tclobjc directory is being included in the compile command for gcc. If it isn't, then add it to the OTHERINCLUDES variable in Makefile.conf and recompile Swarm.
could not find a usable init.tcl in the following directories ..... This probably means that Tcl wasn't install properly
Answer: Assuming you did install Tcl properly, this happens because Swarm can't find libraries that it needs. It does not happen with the archive install method, assuming everything is setup right in Makefile.conf. Binary installation users, however, probably need to create ENVIRONMENT variables.
If it is the only problem in your setup, you can fix without recompiling Swarm by changing your environment variables. First, find those libraries in your file system, then set the environment variable TCL_LIBRARY equal to the path where you found that file. The same should probably be done for TK and BLT. Depending on the sort of Unix you have, you may need to put these in either .profile or .bash_profile. Explicit instructions for environment variables for binary users are shown in "binary installation procedure" above.
Answer: It could be one of two things 1) The dots in the library name or 2) it really is looking for libtk.a. The solution to both problems is to make libtk.a a symbolic link to whatever the latest version of the library happens to be. Archive install users should check ldconfig to find out what their system uses for libtk.a.
Answer: Binary distributions of GCC exist for many platforms. GCC is included in all major Linux distrubutions. Other users should check the GNU sites and the GNU docs for where to find one.
You must apply the patch ./gcc.patch to the gcc source dir
then recompile and install libobjc.a,
then rerun this configure script.
when trying to configure and install libtclobjc.
Answer: The configure script is telling you that your version of gcc won't compile Obj-C. You don't really have to patch gcc. Simply reinstall gcc to include Obj-C (or ask your sysadmin to do it).
Module HashTable.m version 6 doesn't match runtime 7
Compiler (gcc) is out of date
IOT trap (core dumped)
mean?
Answer: This is the warning that the Objective C runtime gives when there's a version skew. Ensure that all the libraries are up-to-date.
collect2: executing gcc: Permission denied
Answer: Something is wrong with your gcc installation. collect2 is a special program used to link Objective C and C++ programs (it's invoked by gcc). Apparently it's trying to invoke gcc, but the gcc it's invoking isn't right. Make sure that the collect2, ld, and gcc you're using are all from the same gcc version. You might want to ask your sysadmin.
Answer: If you can't find sel_get_any_typed_uid() you are probably using gcc 2.7.0: that function was introduced into the Objective-C runtime only in later versions of gcc. The swarm web page has a link to the latest version of the compiler. Upgrade to 2.7.2 or greater and this should go away.
Answer: This appears to be a problem on older SGI systems. The expand command was being used internally by a script called make-h2x, which is used so far to compile the defobj, collections, and activity libraries. expand is a normal Unix command (so I've always thought) I expected to be on all Unix systems. All it does is expand tabs to spaces; if it's not on your system you can get source from ftp://ftp.gnu.org/pub/gnu/textutils-1.22.tar.gz.
make: Fatal error in reader: ../swarm/Makefile.appl, line 12: Unexpected end of line seen.
Answer: I think the problem is you are not using GNU make.
Question: I get the message from ld.so telling me it can't find _XOpenIM (specifically 'undefined procedure') when I try to run the demo apps.
I've tried including a path entry for every X11 library but it still wont run. What could be missing if not the X libraries?
Answer: It sounds more like the version of libX11 that's being dynamically linked is odd. Check that libX11 is correct, and also check that tk and blt were built with the right X11 in mind.
Question: When I start any Objective C program on my Linux machine, I get a floating point exception.
Answer: This is a bizarre bug in Linux libc-5.0.9. The quick fix is to hide the problem by linking all Objective C programs with -lieee (in Swarm, set OTHERLIBS in Makefile.conf to -lieee). Alternatively, upgrade to libc-5.2.18 or newer.
2.2. Development Environment Questions.
2.2.1. Do I have to use Emacs?
Answer: No. Files in an application directory can be edited with any text editor and executed from the command line. Linux-using-Emacs-haters might try a program called Xwpe, that supplies a Borland-style development environment, but it is not tailored to Objective-C.
Answer: For that, you need Emacs, or XEmacs (a competing clone/immitation that newcomers will find easier to use). The .emacs file in your home directory can be edited to make Emacs aware of objective c. The old (deprecated, unsupported) method used a font package called hilit19. Versions 20 of Emacs and XEmacs includes a package called Fontlock. There is built-in support for some color-coding. You have to turn it on, however (see the next question!).
Answer: Emacs is from GNU and integrates tightly with GNU compilers and debuggers.You can do everything with it. A lot of people really like it (once they overcome the initial shock!).
Answer: Yes. Consider the competing program XEmacs. Its authors aim to make it more menu driven, but it also is supposed to comply fully with the Emacs commands and add-on programs. Please note. XEmacs is not the same as the version of Emacs that is X11-aware.
2.2.2. How to get color coding in Emacs (or XEmacs)?
Answer:
1. There have been changes on this topic and I've not sorted them out completely. On Emacs version 20 or XEmacs version 20, Fontlock and color-coding for ObjC is built in. The color coding you get is not, in my opinion, very complete or useful.
One step is to tell the editor which kind of color coding you want. You have to let it know that you want "h" files to use the Obj-C mode of coloring. New versions of Emacs are supposed to know that "m" files are Objective-C files, but I still specify both h and m in the .emacs file in the home directory. Like so:.
(setq auto-mode-alist
(append '(("\\.h$" . objc-mode)
("\\.m$" . objc-mode)
auto-mode-alist))
The use of color is not very good even then, IMHO.
The next step is to add detail to the .emacs file to control the color editing. There has recently (September, 1997) been discussion of code in the .emacs file to use the newer fontlock package with Emacs. Look here to see the thread begin. For XEmacs users, John Eikenberry posted an excellent piece of Lisp Code that can be added to the .emacs file. Click here to see it! That code worked fine for me after fixing a few lines that were split apart by the email program used to post it. Look toward the bottom of the file, you'll see some split words.
Answer: You should edit the .emacs file as described above. Make sure the *.h are appended to the alist and not pre-pended.
2.2.3. Is there a debugger specially designed for Objective-C within Emacs?
Question: Emacs does not seem to understand the subtle differences between C++ and Objective-C. What gives?
Answer: On the Swarm ftp site, there is a binary of GNU debugger that can handle it. Wish I knew more, will look into it!
2.2.4. Where is the Object Browser?
Question: In chapter 9 of Brad Cox and Andrew Novolbiski, Object-Oriented Programming: An Evolutionary Approach, 2ed (1991), they discuss the Stepstone corporation's object browser. I was thinking "I wish I had that!" So?
Answer: There is a lisp addon for Emacs/XEmacs called OO-Browser that provides most of that functionality. I've not installed it with Emacs, but it is included in recent distributions of XEmacs. In the XEmacs version 20.x distribution from the Red Hat site, one finds the OO-Browser already compiled, and it shows up under the Tools menu.
When you choose Tools/OO-Browser, it asks where your environment is, and if you already have a file called OOBR somewhere, you tell it where. If you don't, you can specify directories where OO-browser is supposed to find source files for objects. (While doing this, I got some error messages because some of the implementation files in the swarm 1.0.1 distribution did not have @end statements, but as soon as I fixed that, it was OK!).
After the environment is compiled, it shows a list of all classes. Right click on one, move down to the 'class' header, choose to view ancestors or descendants of that class, and poof, they show in a second buffer. Then choose 'class" and 'features' and all the methods of the chosen class are displayed. Then choose 'features' 'implementors' and you get a list of all classes that have a method of the chosen name. Any time, you can choose to edit a source file from which a class is taken.
I would add these points of caution:
1. The graphical broswer, xoobr, is a dud. Its not installed and if you try to access it, you'll see error messages. It is in the distribution, and can be compiled, but don't bother unless you are using an 8 bit display. Maybe it will get better.
2. I've reconfigured the environment many times, and I've found that if you include both the swarm/include and swarm/src directories, you get a lot of duplicated classes (duh!). So I'm just including swarm/src, along with the directory of the code I'm developing, and I also include the directory where /objc.h is held.
2.2.5. Got any advice about managing a simulation project?
Question: How do I keep track of my simulation project?
Answer 1. Use RCS--Revision Control System. This is included in most Linux distributions, may also be available for specialized Unix systems. RCS is described in various Linux manuals and it has a man page as well. Here is a testimonial:
(Benedikt Stefansson) I've recently discovered the benefits of using RCS to manage my simulation projects. I'm sure that everyone who has spent more than a couple of weeks developing the same simulation knows the feeling that you loose track of all the changes you are making. Tarring and zipping up subsequent "generations" of the project is one solution, but you are still don't have a log of what changes have been made.
RCS basically manages the project and keeps a log of all your changes. The basic premise is that each time you make changes you "check in" files into the program, which first of all means that the files become write protected and second means that RCS prompts you for an update to the log and files away information on the changes made. Then when you want to make more changes you "check out" the respective file. If you want to back up several generations on the file, you can ask RCS to undo the changes.
Another aspect of RCS is that Emacs automatically interfaces with the program, so you can do all the stuff mentioned above directly from the editor. It's no sweat, believe me - I spent all of 30 seconds to understand the concept. I learned all I know about RCS from the Nutshell book, which has 3-4 pages on it, and the man pages on RCS. It is included on the RedHat disks and should be somewhere on most decent Unix systems. (Editor's note: there is no RCS man page, but there are pages for commands ci and co, the commands in RCS that "checkin" a file and "checkout" a file.)
Answer 2. The CVS approach, which SFI developers use.
(Marcus Daniels says this is much more convenient than RCS).
(Glen E.P. Ropella) RCS is an excellent suggestion! But, I'm going to add that *we*, being the Swarm project, use CVS, which is an extension to RCS that allows *concurrent* development by the same user or different users and does a decent job of merging branches and such. So, I would advise CVS; but, especially for a single developer, it only matters that one use *some* revisioning system tool.
Answer 3. Makefile meets tar/gzip (Rick Riolo's alternative to RCS )
Use Makefile to tar-gzip and date things. I use this instead of RCS for little projects for which RCS seems like too much of a pain. I just include a ChangeLog.txt file to record my changes. (yes, yes, I could use RCS, Emacs, etc...something about old dogs comes to mind here...)
Anyway, this does show how to get a date into file names being slung around in a Makefile.
A part of a Makefile might be:
--------------------------------
DATE= `date +%y%m%d`
APPLICATION=sc01
COMPRESS= gzip
UNCOMPRESS= gunzip
Z_EXT= gz
<...snip...>
save:
tar cvf $(APPLICATION).tar *.txt Makefile *.m *.h *.c *.tcl *.pl
$(COMPRESS) $(APPLICATION).tar
if [ ! -d Saves ]; then mkdir Saves; fi
mv -i $(APPLICATION).tar.$(Z_EXT) Saves/$(APPLICATION)-${DATE}.tar.$(Z_EXT)
------------------------------------------------------------------
Entering
# make save
will result in a file Saves/sc01-970825.tar.gz which has all the files
matching the patterns shown on the tar command (assuming I do it when the
computer clock date is 970825).
Note it creates the Saves subdir if it doesn't exist already. Thus I can save some daily work in a dated file, and have dated tar files ready to ship out to other machines.
Also note by using the variables I can copy this Makefile for new projects and then I usually just have to change the APPLICATION variable (well, there is usually an OBJECTS variable too).
2.2.6. I'm new to Unix. Got any FAQ for that?
Answer: No, but you do have to be willing to learn. If you are new to Unix/Linux, you will no doubt have lots of questions. There will be an initial steep learning curve as you learn commands like these. (The symbol # means "Unix Shell Prompt Here"--that's the place where you type commands. Don't type # at the beginning of commands.)
# ls --- get a list of files in the current directory (usage tip: "ls
-la" gives more information!).
# cp x y -- copy file x to filename y (usage tip: you can include directory
names as part of file names).
# mv x y--- move file x to file name y (that erases x).
# rm x --- remove file x (usage tip: type "rm -i x" and watch
what happens!).
# cd dirname ---change to directory "dirname" (cd .. moves you
"back one level" in the file system).
# mkdir dirname --- create a directory called dirname (by default, it is
a subdirectory of the current directory).
After the basics like that, you'll come across more interesting problems and this FAQ is supposed to help you address them.
Warning about brands and shells. There are several versions of the Unix operating system. You can use Linux, DEC UNIX, HP, etc. There are some subtle differences, especially where free software from GNU is concerned. GNU stuff (like "make" or "gzip") is available as standard on some systems, not others. Also, the shell that is in use--Bourne Shell, Korn Shell, bash, C-shell, etc.--will affect some commands, particularly in the creation of environment variables in the user's setup. So, you have to be self-aware enough to find out what version of UNIX you have and what shell you are using.
Answer: Try typing
# man commandname (for example, type "man ls" to see about file lists).
If the built-in manual system called "man" has an entry for "commandname," you can read it. Some man pages are helpful, many are hopelessly abstract and discouraging to a new comer. There are many manuals you can buy for Unix beginners.
Also, there is the GNU info program. GNU folks have tried to convince everyone that man pages are bad and that the info format is good (that's why so many man pages say "this is old, use the GNU info page to get the real information). The command "info" will bring up the GNU information pages. If you have Emacs or XEmacs installed, you have easy access to info.
Answer: The compiler probably has to be told in which directory it will find that file. How you specify the location may depend on the program you are compiling. It may be set in an environment variable, a makefile, or a gcc command option.
Now, how do you find out where the file is on your system? If you have that file in your file system, commands like "find" and "whereis" and "locate" can be used to find out where they are.
1) The find command
Example (Rick Riolo): Type this (don't type #. Remember, that's the shell prompt!):
# find / -name libX11.a -print
find: cannot open /dev/diag
/usr/lib/X11R5/libX11.a ...
The above command looks at all the directories on the system below / (root dir), but you could also look in specified subtrees, eg find /usr -name "libX11.a" -print. (Editor's Note: On my system, the -print option is not necessary. It is assumed by default. To speed up the search, add the option -xdev before -name. This confines the search to the current file system, meaning that DOS drives that may be mounted are ignored. If one is unsure of a file's full name, add asterixes around the part that you are searching for, such as "*name*". This is regular expression syntax. The "cannot open" messages are not errors. They are a result from some directories refusing to allow themselves to be inspected because the user does not have permission to open them. )
To find the all versions of X11 library files, such as libX11.a, one can use regular expression syntax, as in:
#find / -name "libX11.*" -print
find: cannot open /dev/diag
find: cannot open /usr/lib/X11/fonts/ifo.st/typefaces
/usr/lib/X11R4/libX11.sl
/usr/lib/X11R5/libX11.sl
/usr/lib/X11R5/libX11.a
(Of course, the response on your system will depend on what file you have! :) )
2) The whereis command (Ginger Booth):
Type this:
# whereis X11
X11: /usr/bin/X11 /usr/local/bin/X11 /usr/local/lib/X11 /usr/contrib/bin/X11 /usr/contrib/lib/X11
3) The locate command (Jan Kreft):
I use locate instead of find because I don't have to remember any syntax and locate is much faster so you don't waste time if you made a typo and have to repeat the search.
locate .gif will print all the gif pictures you have on your disk in an instant. Very nice.
To be able to use locate, cron.daily (called by the cron daemon) must be configured to update the locatedb database once per day. You must have sys adm rights to do this. The details may be rather system specific.
Answer: The first option is to pipe the results into less or more by adding |less (or |more) on the end of your command. (less is the gnu version of more, so some systems have less and more, but HP or DEC systems might have only more). This will make sure it prints one screen at a time. The more sophisticated and easy answer is to pipe the results to the grep program, which will then pluck out all lines that have a certain word.
Example (Ginger Booth): Pipe grep is an approach to finding arbitrary
things by string match.
# ps -ef | grep netscape
ginger 24257 24088 0 13:10:26 ttyp4 0:04 netscape -install
# ls -al /tmp | grep "Aug 25"
drwxrwxrwx 3 root root 37888 Aug 25 13:40 .
-rw-r--r-- 1 ginger staff 0 Aug 25 10:17 SPCBAAa23733
-rw------- 1 ginger staff 366 Aug 25 10:46 jea23850
-rwx------ 1 ginger staff 771475 Aug 25 13:37 mbox.ginger
-rw------- 1 ginger staff 1724 Aug 25 13:40 snd.24249
Answer (Jan Kreft): Use filters such as grep, cut, paste, uniq, join
Example: Here are two dump files to start with:
file 1: bac.log
min 0 vol 0.399996
Time: 0
min 0.1 vol 0.40122
Time: 1
min 0.2 vol 0.402444
Time: 2
file 2: bac.biomassGraph
0 231.997666 0.805550
1 232.707558 1.611180
2 233.417510 2.418453
Aim: Only the values for min and vol of the first file and the second column of the second file are to be imported into a spreadsheet.
Step 1: grep file 1 for lines containing min, then cut fields 2 and 4 and output into file tmp.
grep min bac.log | cut -f2,4 -d" " > tmp
Step 2: cut field 2 from file 2 and output into file tmp2.
cut -f2 -d" " bac.biomassGraph > tmp2
Step 3: combine corresponding lines horizontally from files tmp and tmp2 and output into file tmp3.
paste -d" " tmp tmp2 > tmp3
Result: file tmp3
0 0.399996 231.997666
0.1 0.40122 232.707558
0.2 0.402444 233.417510
This file can be imported into Excel etc. Note that the option -d" " was used throughout to produce blanks as delimiters (memo d=delimiter). The option -f was used to specify the field numbers for cut (memo f=field). Use command --help to remind you of the syntax and options and man command for more detailed explanations.
Answer: Use alias. Alias lines can be added to the Unix startup files (.bashrc, .cshrc, or .profile, depending). If you find yourself typing a command all the time, create an alias so that you can type just a few letters and the machine thinks you typed the whole long command.
Example (Rick Riolo): I find alias very useful for making shortcuts for stuff I do all the time.
alias pgrep 'ps -ef | grep '
alias hgrep 'history | grep '
The first is to find out whether some program is running (eg to find out what the process id is so I can kill it!):
# pgrep emacs
1085 950 2 21:58:43 ttyp7 0:00 grep emacs
1009 946 0 08:58:37 ttyp6 0:07 emacs Notes-Series3.txt
The second is to find out exactly what command I entered a while ago (without have to scroll or arrow-key back...this depends on using a shell like tcsh which has a history command:)
#hgrep wc
191 15:20 wc Classifier/BasicCollection.m
The number on the left is the command number in the history list, next is the time of execution, and then the command.
Answer: Consider the "diff" command. It will compare files, and also can execute patches (see the manual...)!
Example: (Ginger Booth) Compare (few) source code changes between versions.
An alias command can be added to the .cshrc file like this:
alias cmf '/usr/local/bin/diff -b --ignore-space-change --minimal -s -C 1 '
so the command "cmf" activates "diff" with all those options. Please consult "man diff" for your options. If your shell is different, a slightly different format for the alias command may be needed. Consult documentation for your shell.
For example, to compare two files, source/Diffuse.h and fromjanwith/Diffuse.h, you type
# cmf source/Diffuse.h fromjanwith/Diffuse.h
*** source/Diffuse.h Mon Jul 21 13:25:51 1997 --- fromjanwith/Diffuse.h
Mon Aug 18 13:13:12 1997 ***************
*** 2,4 ****
! #import <objectbase.h>
#import "Base.h"
--- 2,4 ----
! #include <objectbase.h>
#import "Base.h"
***************
*** 63,64 ****
--- 63,65 ----
-copymom; // self=mom, returns copy
+ -copyTo: (DiffuseQuad*) newbie; // no alloc
-addSiteParams;
***************
*** 102,103 ****
--- 103,105 ----
-(int) getCellsY;
+ -(int) getSubSteps;
-setSubSteps: (int) i;
It's hard to get really legible output out of this, but the above says I changed (!) an import to an include (lines 2-4 shown for context), and added (+) two routines (lines 64 and 104 in the 2nd file.)
Answer: Use wc -l.
Example: (Ginger Booth) The command wc -l will count lines of code, as a rough progress measure or version compare.
# wc -l source/*.m
127 source/Base.m
416 source/Carnie.m ...
39 source/main.m
14394 total
Answer 1. This file is a GNU-zipped tarball! The program "tar" crams a bunch of files into one file (with a tar suffix). GNU-zip is a compression program for UNIX (comparable to PKzip for DOS). First, figure out where you want to unpack the archive. Move it in that directory. Then un-gzip, then untar. Type
#gzip -d filename.tar.gz
(The commands "gzip -d" and "gunzip" are the same!)
and then (if you trust the package, proceed. Otherwise, look at answer 3 below)
#tar -xvf filename.tar.gz
This creates directories if they are called for in the tar file, or writes files into existing directories if you have unzipped the files from a directory where the system can find the right subdirectories.
If you are fond of pipes, you can do this in one shot (notice we show gunzip in placeof gzip -d for variety).
# gunzip -c filename.tar.gz | tar -xvf -
The switch -c sends the output of the unzipping process to the command after the pipe, where it is picked up because the "stdin" is called for by the dash after xvf.
Answer 2. Simple one step command available on some Unix systems (not AIX, DEC, or HP if they are "out of the box" versions):
# tar -xvzf <filename.tar.gz>
("z" means gunzip the file on the fly, "x" means eXtract, "f" says you're going to give a Filename; and "v" says be Verbose, i.e. send a representation of the contents to stdout.)
This works for files with suffix .tgz as well.
Answer 3. For safety, before untarring consider getting a listing of what is in there. It could be that the person who created the tarball did a crappy job and they are going to dump files all over your hard disk, so WATCH OUT. If you have already unzipped a file, you can use the tar command to view a list of the contents by typing:
# tar -tvf <filemame.tar> | less
This gives a file list.
If your tar allows the z switch, then you can look at the contents of the tarball without unzipping it. This is the command:
#tar -tvfz filename.tar.gz | less
(If you don't have the "less" program, use "more")
Answer 1. Tar can create an archive for you. To put all the files and directories under /users/someone/ into a tar archive called bozos.tar, type
tar cvf bozos.tar /users/someone/*
The switch "c" creates an archive', "v" gives back
a verbose report on the process, and "f" tells tar that a file
list is following. The * means that every file and every subdirectory will
recursively be put into the tar file. If you only want to get the files
with extension jpg, replace * with *.jpg. Caution, do not type this:
# tar cvf filename.tar *
That will make the tar program try to put filename.tar into itself! It doesn't cause a crash, just a weird tar file. That's why it is best to put in absolute paths (such as /users/someone/* when you create tar archives.
Answer 2. Notice that some people put a - sign before the switches in a tar command and others don't. They used to be required on most unix systems, now they aren't, but most commands work either way. Rather than be consistent, I've left most of the examples provided by others in the style they chose.
Answer 3. (Rick Riolo) You can extract particular files without dumping them all. If you want a file called Tribble.m that is in a directory called "source" in a tar file, do this:
tar xvf tarfile.tar source/Tribble.m
Answer 4. As an example of how tar can be used to ship files to a colleague, consider this example (Ginger Booth). Suppose a directory called backups exists. In backups, we want to create a tarball called prestats.tar. Tar is going to get files from a directory called source (in this case, the files are Base.m and Critter.m).
# tar -cvf backups/prestats.tar source
a source/Base.m 8 blocks
a source/Critter.m 33 blocks
...
# ls backups
prestats.tar
...
# gzip backups/prestats.tar
# cd backups
/users/ginger/Sgecko/gecko/backups
# ls
prestats.tar.gz
...
Now, you've got it tarred and gzipped. If you wanted to reverse the process, do this:
# gunzip prestats.tar.gz
# tar -xvf prestats.tar
x source/Base.m, 3674 bytes, 8 tape blocks
x source/Critter.m, 16419 bytes, 33 tape blocks ...
#114: ls
prestats.tar source
Answer: That's what I thought too. Then I heard about ldconfig. On my Linux machine, it works like this. The ldconfig program runs every time you start your machine, and it maintains a listing of libraries that are available for programs to use. If you don't restart, chances are the library you installed is not yet known to your system. If you don't want to restart, find out where your system has ldconfig, and execute it. In my Red Hat Linux system, it is in the directory /sbin, so I type
# /sbin/ldconfig
(Actually, before I do that, I look in the file called /etc/ld.conf to make sure the directory into which I added the library is included in the ldconfig's search path. If it is not in there, I add it.)
And then to get a listing of the files the libraries the system knows it has, type
# /sbin/ldconfig -p
Look it over. On the left are "stub" names of libraries. A library will have a version name like 2.3, where 2 is the "major version" and 3 is the "minor version." In the library, there is a hardcoded "stub" which indicates that this library will respond to a request for a certain library name, usually it cuts off the minor version number. In the listing, on the right are stub names that provide the needed information or there are symbolic links to files that can do it. If a system is looking for Motif library version 1.2.4, for example, the ldconfig program will look at the file libXm.so.1.2.4, and it will do two things. First, it sees the "stub" name, the major version name, is libXm.so.1, and it will create a link that points from libXm.so.1 to libXm.so.1.2. That's from the major name to the first minor version name. Then ldconfig will create a symbolic link in the directory where it finds that library from libXm.so.1.2 to libXm.so.1.2.4. So, any program that calls for the Motif library should ask for the major version libXm.so.1, and then the ldconfig knows it has that, follows the links to libXm.so.1.2.4.
Answer: Gee, maybe you installed the wrong version of the library! If you are a Linux user, use the ldd program to find out what version your program is looking for. "ldd program-name" causes a list of libraries that the program uses to be printed on the screen. Other Unix systems have similar facilities.
Example: Netscape for linux includes a version dynamically linked to Motif library 1.2.4. That file is called libXm.so.1.2.4, but when Netscape-dynamic is executed, it asks for libXm.so.1.2, and the linker figures out what it needs. The output from ldd on a Red Hat Linux 4.2 machine looks like this:
libXm.so.1.2 => /usr/X11R6/lib/libXm.so.1.2.4
libXt.so.6 => /usr/X11R6/lib/libXt.so.6.0
libSM.so.6 => /usr/X11R6/lib/libSM.so.6.0
libICE.so.6 => /usr/X11R6/lib/libICE.so.6.0
libXmu.so.6 => /usr/X11R6/lib/libXmu.so.6.0
libXpm.so.4 => /usr/X11R6/lib/libXpm.so.4.9
libXext.so.6 => /usr/X11R6/lib/libXext.so.6.1
libX11.so.6 => /usr/X11R6/lib/libX11.so.6.1
libdl.so.1 => /lib/libdl.so.1.7.14
libc.so.5 => /lib/libc.so.5.3.12
libg++.so.27 => /usr/lib/libg++.so.27.1.4
libstdc++.so.27 => /usr/lib/libstdc++.so.27.1.4
libm.so.5 => /lib/libm.so.5.0.6
Question: How does one position the Swarm windows?
Answer: In the "old days" before Swarm 1.0.5, this was a problem. Now, there is a handy feature. When the user creates a display window, say call it "my window," use the command SET_WINDOW_GEOMETRY_RECORD_NAME(myWindow); before the createEnd statement. For example, here it is in heatbugs-1.0.2 in the main.m file.
theTopLevelSwarm = [HeatbugObserverSwarm createBegin: globalZone]; SET_WINDOW_GEOMETRY_RECORD_NAME (theTopLevelSwarm); theTopLevelSwarm = [theTopLevelSwarm createEnd];
When the swarm program starts, in addition to the START and STOP buttons, there is a SAVE button, which records the window positions in a hidden folder. Next time the application starts, the window positions are remembered. The windows appear in whatever "workspace" holds the xterm from which the application is started. (Its not yet possible, on a system using a multi-workspace display, to divide the windows up among the workspaces.)
Question: If I open a window to look at parameters (for the HeatbugModelSwarm or HeatbugObserverSwarm), and then quit that window, then when I click on "go" to run the program, I get an error message repeated over and over until I quit the program altogether.
Answer: That means you are running an old version of Swarm, most likely. Newer Swarm versions include "enableDestroyNotification:..." that can close down programs if you close the window with a Window manager, rather than the "quit" button on the control panel. Older Swarm versions had no such feature, and so the programs did not close nicely when you killed their X display windows.
Question: I get a "ResourceAvailability" or "Insufficient Resources" warning when I start a Swarm application. What's it mean and how do I fix it?
Answers:
for (i = 0; i < 64; i++)<newline> [colormap setColor: i ToRed: (double)i / 63.0 Green: 0 Blue: 0]; to
for (i=0;i<16;i++)<newline> [colormap setColor: i ToName: "black"];<newline> for (i=16;i<32;i++)<newline> [colormap setColor: i ToName: "blue"];<newline> for (i=32;i<48;i++)<newline> [colormap setColor: i ToName: "orange"];<newline> for (i=48;i<64;i++)<newline> [colormap setColor: i ToName: "red"]; This will replace the Red gradient that Heatbugs normally uses with four colors. The coolest color is black and the warmest is red. Using this method, you can see where the heat is, but not as precisely.
startx -- -bpp XX where XX would be 16 or 32. If your system does not support 16 or 32 an error message will be piped to the stdout.
Apparently, in some versions of X, the function XAllocNamedColor does not handle the hex string passed to it by setColor:ToName: (from setColor:ToRed: Green:Blue). XColormap.m can be changed to use XParseColor() before it calls XAllocColor().
Question: On the test "graph", I pick on a "config" button and get a lovely blank window. Testing the swarm apps, I also get a lot of charmingly blank windows.
Answer: This is a BLT/Tk problem. Make sure you're using a version of BLT that works with the current version of Tk.
It also pays to double check that all versions of blt, tk, tcl you're using, linking, including are correct - I once spent eight hours chasing down a bug that was because of a version mismatch. If you have several versions of tk installed, it can be hard to find the right version.
(you can build a quick test of BLT by doing "make testGraph" in the support/tkobjc directory. It should make a program that draws a little random time series graph.)
Question: When I run it my hpux displaying on its own screen, some of the graphs data parts are about 0.5cm wide---unreadable, while the part that displays the correspondence between colors (the squares or the lines) and variables are readable.
Answer: There are two possible contributors to this problem:
It is thought that rebuilding the GNU C compiler (gcc) using the GNU Assembler (gas) instead of the system assembler and replacing "rint(x)" calls with "floor(x + 0.5)" will fix this problem. The real source of the phenomenon is unknown.
Question: I just noticed that the graphs for the stock market simulation don't scale right if you ask to plot all 16 predictors - the graphs show about a centimeter of plots and about 5 meters of predictor names and their associated colors.
Answer: If you are still having trouble with the graphs, try only plotting 7 or less predictors (choose in the initial parameter window...)
Question: Mousetrap still doesn't do anything when I hit go (except it does become the top cycle burner, so its doing something...). I see the red and black display and two graphs which run (x) from -10 to 10. But the red/black pattern doesn't change and nothing gets drawn on the graphs. (I've waited about 5 minutes or so.) What should I be see happening?
Answer: What you are probably seeing in the graphical display is the end result when all is done, which is a pattern of red and black that isn't changing. At the end of the run the message "All the balls have landed!" will appear in the window from which you started the app.
Question: I get the following run-time error after running for a long period of time:
(Tcl -eval:) can't allocate data vector while executing<newline> ".wFrame@0x4005e210.wBLTGraph@0x4005dfe0 element append GraphElement<newline> x40062128 { 6090 585 }"<newline> while evaluating: .wFrame@0x4005e210.wBLTGraph@0x4005dfe0 element appe<newline> nd GraphElement@0x40062128 { 6090 585 };<newline> *** event raised for error: OutOfMemory<newline> No more memory available from the system. Value of sbrk: <newline> *** execution terminating due to error<newline> IOT trap (core dumped) Answer: Unresolved. Please send ideas to <swarm-support@santafe.edu>
Question: I tried to compile "heatbugs" for a test. It appeared to compile correctly, but when I run the demo, I receive a continuous stream of error messages of the form:
(Tcl -eval:) invalid command name "blt_bitmap"
Answer: This means you're using a version of Swarm that does not work with BLT v2.0 or greater. Either get a newer release from the Swarm Web site, or use an earlier version of BLT.
Question: How hard is it to get my points back out of a BLTGraph? Looking at BLTGraph.m, we've got:
-addX: (double) x Y: (double) y {<newline> [globalTkInterp eval: "%s element append %s { %g %g }";,<newline> [ownerGraph getWidgetName], name, x, y];<newline> return self;<newline> }<newline> <newline> -resetData {<newline> [globalTkInterp eval: "%s element configure %s -data { };",<newline> [ownerGraph getWidgetName], name];<newline> return self;<newline> }
And I want something like
-getDataI: (int) i X: (double*) x Y: (double*) y;
Or a pointer to an array of points, if there's an array of points, or whatever.
Answer: The BLT command for this is
.g element configure e -data
i.e., like "resetData" but don't give it the actual data. The return is:
-data elemData Data {} {0.0 0.0 1.0 3.0 2.0 5.0}
Question: Does BLTGraph actually communicate with Tcl/Tk through these string lists? Does Swarm communicate with BLTGraph solely through the tkinterp by necessity or policy?
Answer: Unresolved. Please send ideas to <swarm-support@santafe.edu>
Question: Where is the problem with the following ActiveGraph object creation?
BLTGraph *mflowGraph;<newline> GraphElement *materialElement;<newline> ActiveGraph *mflowGrapher;<newline> <newline> StatusM *statusM;<newline> <newline> mflowGraph = [BLTGraph create: globalZone];<newline> [mflowGraph title: "Matrial Flow vs. Time"];<newline> [mflowGraph axisLabelsX: "time" Y: "Material Flow"];<newline> [mflowGraph setWidth: 400 Height: 250];<newline> <newline> materialElement = [mflowGraph createElement];<newline> [materialElement setLabel: "Shipment of Products"];<newline> [materialElement setColor: "red"];<newline> <newline> mflowGrapher = [ActiveGraph createBegin: globalZone];<newline> [mflowGrapher setElement: materialElement];<newline> [mflowGrapher setDataFeed: statusM]; <newline> [mflowGrapher setIntAccessor: M(getM)]; <newline> mflowGrapher = [mflowGrapher createEnd];<newline> [mflowGraph pack];<newline>
The StatusM object is defined as:
@interface StatusM: SwarmObject
-(int) getM;
@end
@implementation StatusM
-(int) getM {
return partSINK;
}
@end
Answer: In Objective-C declaring a variable to be of a particular class does not automatically generate an instance of that class in the variable. In other words the variable is initialized to nil.
BLTGraph *mflowGraph;
GraphElement *materialElement;
ActiveGraph *mflowGrapher;
StatusM *statusM;
And therefore the statement:
[mflowGrapher setDataFeed: statusM];
is setting the DataFeed on mflowGrapher to nil. The grapher therefore complains since it does not have a valid feed object.
You would therefore need to create the object statusM before giving it as a dataFeed to the grapher by (for example) inserting the statement:
statusM = [StatusM create: globalZone];
between the declaration of statusM and its use...
For the record, local variables are not initialized to any particular value (as opposed to global or static variables) and since statusM is a local variable it would normally not be initialized to nil (and certainly not initialized to any valid object instance).
In this particular case, though, we were lucky in that the value of statusM was nil which caused the grapher to believe you had never set the feed at all... Otherwise it would have attempted to talk to the feed which would have failed too because as mentioned above, the statusM variable did not yet contain a reference to a valid object (since none was created).
Question: Heatbugs dies. Swarm make'd without a whimper, so did the other apps (mousetrap and market), and they executed fine. But heatbugs crashed with "Illegal instruction (core dumped)". Gdb: no backtrace. So I tossed in some printfs, and found that heatbugs executes fine, after commenting out each of 3 calls to createProbeDisplayFor: . Why would Heatbugs crash due to createProbeDisplyaFor?
Answer: Unresolved. Please send ideas to <swarm-support@santafe.edu>
Question: Make doesn't work because the linker cannot find srandom, random, or rint on HPUX.
Answer: HPUX apparently has more than one version of libm.a, one in /lib and one in /lib/pa1.1. The version of libm.a in /lib/pa1.1 is, supposedly, a superset of the one in /lib plus utilities that are non-ANSI. To fix this, modify the Makefile.conf file to add "-L/lib/pa1.1" to the OTHERLIBDIR variable.
Calls to random and srandom functions should be avoided. We're taking them out of the Swarm kernel; but, some calls may still remain. The OO Random library should be used if possible.
And rint() can probably be replaced with floor(x+0.5) in most cases. Rint() does exist in HPUX /lib/pa1.1; but, there is a claim that there may be a bug in it.
Question: Why would Make fail when RE-compiling a Swarm App without having made any changes to the code?
Answer: It's possible that one of the library's symbol tables has to be regenerated. To find out which one (if any) is messed up, look for a warning in the make output that looks something like the following:
ld: rhome/sun32/mlg/swarm/swarm/../libtclobjc-1.1b2/lib/libtclobjc.a: warning: table of contents for archive is out of date; rerun ranlib(1)/
Then run ranlib on the offending library (libtclobjc.a in this example).
Answer: Unresolved. Please send ideas to <swarm-support@santafe.edu>
make: *** No rule to make target `-ltkobjc', needed by `market'. Stop.
Answer: This indicates that make didn't find "libtkobjc.a" (or whatever the case may be for your error) and it doesn't know how to create it itself. Either your "libtkobjc.a" was not installed or did not get created properly. In this example, "libtkobjc.a" is a part of Swarm and you can check whether the make was successful by changing to directory
$SWARMHOME/src/tkobjc
and type make.
Undefined first referenced<newline> symbol in file<newline> Tcl_GetVar /net/echelon/opt8/swarm/swarm/lib/libtkobjc.a(TkExtra.o)<newline> Tk_DoOneEvent /net/echelon/opt8/swarm/swarm/lib/libsimtools.a(SimParameters.o)<newline> Blt_Init /net/echelon/opt8/swarm/swarm/lib/libtkobjc.a(TkExtra.o)
Answer: The libraries were probably not included in the link command, and if they were, then the linker search path was probably not amended with the directory in which the given library should be found. For example, the above error was generated with the linker command:
gcc -L/net/echelon/opt8/swarm/swarm/lib \
-L/net/echelon/opt8/swarm/tcl7.4/lib \
-L/net/echelon/opt8/swarm/tk4.0/lib \
-L/usr/openwin/lib -L/net/echelon/opt8/swarm/blt-1.9/library \
-L/net/echelon/opt8/swarm/libtclobjc-1.1b2 -L/usr/local/lib -g \
-o heatbugs Heatbug.o HeatSpace.o main.o \
-lspace -lsimtools -ltkobjc -lactivity -lrandom -lobjectbase \
-lcollections -ldefobj -lX11 -lobjc -lm
The following missing flags probably caused the above error:
-ltclobjc -lBLT -ltk -ltcl
The libraries included on the linker command library are customizable in Swarm via the OTHERLIBS variable in Makefile.conf in the
$SWARMHOME
directory.
Answer: Check the order in which the libraries are specified in the link command. For example, if the object bob() in libxyz.a is referenced by dave() in libpqr.a and the linker command make generated lists -lxyz -lpqr, such that the libxyz.a is searched first, then the linker may not find bob because it's already searched libxyz.a when it gets to the reference to bob().
In other words: Order matters. Try rearranging your library references in the link command. For Swarm apps, it's a good idea to preserve the order in the Makefile.conf.
Question: I'm getting warnings about undefined functions: printf, malloc, calloc, etc. when I compile. What's wrong?
Answer: In some of the include files used by GCC, the prototypes for some of the standard functions are placed in a C preprocessor
#ifdef block relying on the __USE_FIXED_PROTOTYPES__ macro. If you use one of these functions, you can either put
#define __USE_FIXED_PROTOTYPES__
in your source code file, or you can use the -e "EXTRACFLAGS=-D__USE_FIXED_PROTOTYPES__" flag on the make command.
Question: I get a segmentation fault when I run heatbugs. When using gdb to debug this, it says that the fault occurred at line 104 of MessageProbeWidget.m. What do I do?
Answer: This has been fixed in the swarm source code. Please get a newer version.