Friday, January 29, 2010

Using taskfs from Linux.

The plan9port is allowing using synthetic inferno filesystems from Linux systems. I have used this to simplify testing of inferno setup. This blog will explain how this test-setup is created.

Firstly, you will need to install plan9port by downloading plan9port.tgz. Detailed installation instructions are available here , but I will repeat here what I have done for this.

mkdir for_plan9
cd for_plan9
wget http://swtch.com/plan9port/plan9port.tgz
tar -xvf plan9Port.tgz
cd plan9
./INSTALL

You will need to add few paths related to plan9port to your path. Once that is done, you can mount exported inferno filesystem using 9pfuse. Once mounted, these files can be manipulated using simple linux tools. For unmounting these filesystem you can use fusermount with -u option.

Now, for test purposes, I have two inferno instances running on inferno-test host using separate port numbers, and one is running locally. For automatically mounting these remote ends and export local taskfs startup script of inferno is used. This startup script option works well with Linux emu, but may not work on other platform, so check before using. It can be used as emu -v . Following is the script which starts my inferno with taskfs.

$ cat runInfernoScript.sh
#!/bin/bash
BASEDIR=/home/pravin/projects/inferno/pravin/
INSTALL_DIR=inferno-rtasks
$BASEDIR/$INSTALL_DIR/Linux/386/bin/emu -r $BASEDIR/$INSTALL_DIR/ -v /dis/sh.dis /dis/rmount.sh

here, /dis/sh.dis is responsible for inferno shell and /dis/rmount.sh is my script which does all the initialization I need. Here is the content of it.

echo exporting /task
styxlisten -A tcp!*!5555 export /task
echo Mounting 6666 on host1
mount -A tcp!9.3.61.180!6666 /remote/host1
echo Mounting 6667 on host2
mount -A tcp!9.3.61.180!6667 /remote/host2
echo Mounting from localhost:6666 on host3
mount -A tcp!127.0.0.1!6669 /remote/host3
echo Mounting from localhost:6667 on host4
mount -A tcp!127.0.0.1!6670 /remote/host4
pause # this will stop emu from quiting immediately.
echo You should never see this line in output.

Command pause on second last line makes sure that emu will not terminate after executing startup script.

Now, lets go to Linux shell, and execute following script.

#!/bin/bash
set -e
cmd='hostname'
9pfuse localhost:5555 mpoint
cd mpoint/remote
res=`cat clone`
echo "Taskfs resource allocated is $res"
cd $res
echo res 4 > ctl
echo exec $cmd > ctl
echo "Showing output bellow --------------------"
cat stdio

echo "Showing stderr bellow --------------------"
cat stderr


echo "Showing status bellow --------------------"
cat status

cd ..
cd ..
cd ..
fusermount -u mpoint
echo "done -------------------------------------"

Above code does following.

    Mount taskfs using 9pfuse.
    Reserve 4 remote resources.
    Execute hostname command on them.
    Get the output from stdio.
    Show any errors from stderr.
    Show the information related to the status of all remote nodes.
    Release taskfs by unmounting it using fusermount.


And here is the output of running this script.

$ ./testRun.sh
Taskfs resource allocated is 0
Showing output bellow --------------------
inferno-test
BlackPearl
inferno-test
inferno-test
Showing stderr bellow --------------------
Showing status bellow --------------------
cmd/0 5 Done /home/pravin/inferno/host2//ericvh-brasil/ hostname
cmd/1 5 Done /home/pravin/projects/inferno/ericvh//ericvh-brasil/ hostname
cmd/1 5 Done /home/pravin/inferno/ericvh//ericvh-brasil/ hostname
cmd/1 5 Done /home/pravin/inferno/host2//ericvh-brasil/ hostname
done -------------------------------------

You can see from the output that, out of 4 remote resources requested, 3 requests went to two inferno instances running on same host and one was allocated to single inferno instance running on other host. Only one instance was used twice, all other inferno instances were used only once.

So, this is first working proof that taskfs can be used to distribute jobs to multiple nocdes. And this can be done simply by mounting taskfs and using it directly.

Thursday, January 28, 2010

Taskfs with one-to-many support

As promised the 123f4c23dd has support for one to many task execution. In simple words, it means that if you mount multiple remote resources, any command issued will be executed on all remote resources with input provided to all of those nodes. The stdio, stderr and status files will provide contents from all remote locations in merged format.

At implementation level, once the resources are reserved, any write command on taskfs resource is repeated on corresponding files on all remote nodes. Similarly reads on taskfs files propagated with data from all remote node files.

Following are two scenarios of one to many taskfs usages.

; cd /task/remote
; cat clone
0;
; cd 0
; cat status
cat: read error: No remote reservation done
; mount -A tcp!127.0.0.1!6666 /remote/host1
; echo res 2 > ctl
; cat status
cmd/0 1 Closed /home/pravin/projects/inferno/ericvh//ericvh-brasil/ ''
cmd/1 1 Closed /home/pravin/projects/inferno/ericvh//ericvh-brasil/ ''
; echo exec pwd > ctl
; cat stdio
/home/pravin/projects/inferno/ericvh/ericvh-brasil
/home/pravin/projects/inferno/ericvh/ericvh-brasil

Above is the first scenario of reserving two remote resources when there is only one remote host is available, namely host1. Request for two resources is satisfied by creating two separate sessions on same remote host. As a result of this, output of pwd command is same on both sessions, which can be seen from contents of stdio.


; cd /task/remote
; cat clone
0;
; cd 0
; mount -A tcp!127.0.0.1!6666 /remote/host1
; mount -A tcp!127.0.0.1!6667 /remote/host2
; echo res 2 > ctl
; cat status
cmd/0 1 Closed /usr/share/lxr/source/inferno//ericvh-brasil/ ''
cmd/2 1 Closed /home/pravin/projects/inferno/ericvh//ericvh-brasil/ ''
; echo exec pwd > ctl
; cat stdio
/usr/share/lxr/source/inferno/ericvh-brasil
/home/pravin/projects/inferno/ericvh/ericvh-brasil
;

In this second scenario, there are two remote nodes. Namely host1 and host2. Reserving two remote resources end put reserving one session on each of this remote node. As a result, the output in stdio shows different result for both nodes.

Wednesday, January 27, 2010

Support for explicit reservation in taskfs

Here is the 4cff22b0ac revision which supports res command on ctl file. This involves few fundamental changes in how reservation is done. Remote reservation is moved from clone file read to ctl res command. The ctl exec command and stdio, status, stderr files are modified to to verify if remote resources are allocated or not before doing anything. If no remote resources are reserved, then No remote reservation done error is returned.

Currently, remote reservation of only one node is supported. Next revision of code will hopefully support multiple remote reservations and executions.

Following is the demo execution of how one can use this.

; cd /task/remote
; cat clone
0; cd 0
; echo exec pwd > ctl
echo: write error: No remote reservation done
; cat status
cat: read error: No remote reservation done
; echo res 1 > ctl
echo: write error: No remote resources available
; mount -A tcp!127.0.0.1!6666 /remote/host2
; echo res 1 > ctl
; echo exec pwd > ctl
; cat stdio
/home/pravin/projects/inferno/ericvh/ericvh-brasil
; cat status
cmd/0 3 Done /home/pravin/projects/inferno/ericvh//ericvh-brasil/ pwd
; cat stderr
;


Above output shows how ctl exec command and status behaves
before and after ctl res command.

Monday, January 25, 2010

Taskfs with one to one mapping

Taskfs is now properly working with one to one mapping between Taskfs resource and single remote node. Current implementation of Taskfs allows users to mount multiple remote nodes inside /remote as subdirectory with any name. Taskfs will find all valid remote resources, and reserve one for current job execution. Currently scheduling between remote resources is not implemented as no information about load or remote session is available. But once this information is available, making informed scheduling decision is trivial modification to this code. Also, the validity of remote resource is checked by presence of local/clone file. The revision no. of this code is 5476644bd5.

Next target for Taskfs is support for one to many mapping. This will allow single Taskfs resource to manage and utilize multiple remote resources. By this mechanism, by sending single command to ctl file in Taskfs resource, that command can be executed on multiple remote resources simultaneously, and output will be automatically aggregated back into single stdout file of Taskfs resource.

Following is the demo of Taskfs with one to one mapping.


; cd /task/remote
; cat clone
cat: cannot open clone: Remote resources not present at [/remote]
; mount -A tcp!127.0.0.1!6666 /remote/host1
; cat clone
0; cd 0
; echo exec pwd > ctl
; cat stdio
/home/pravin/projects/inferno/ericvh/ericvh-brasil
; cd ..
; mount -A tcp!127.0.0.1!6667 /remote/host2
; cat clone
0; cd 0
; echo exec pwd > ctl
; cat stdio
/usr/share/lxr/source/inferno/ericvh-brasil
;


This demo shows that the output of two pwd commands is different. This is because taskfs selected host1 node when only one remote node was available. But it selected host2 node when there were two nodes available.

Friday, January 15, 2010

taskfs : executing commands on remote host

The first working version of taskfs is complete. This version allows user to execute host command on host which is running remote inferno instance. And these commands are executed using filesystem interface. Revision 49060921b4 of inferno-rtasks implement these modifications.

Here are the steps about how to run and test it.

Firstly, you will need to run one remote instance of inferno with devcmd2 support. ericvh-brasil has this support. here are the commands used for this remote instance.

$ pwd
/home/pravin/projects/inferno/ericvh/ericvh-brasil/
$ ./Linux/386/bin/emu -r `pwd`
; styxlisten -A tcp!*!6666 export /cmd2


Now, following are the commands used with inferno-rtasks.


$ pwd
/usr/share/lxr/source/inferno/inferno-rtasks/
$ ./Linux/386/bin/emu -r `pwd`
; cd /task/remote
; cat clone
cat: cannot open clone: '/remote/local' does not exist
; mount -A tcp!127.0.0.1!6666 /remote
; cat clone
0;
; cd 0
; ls
args
ctl
env
ns
status
stderr
stdin
stdio
stdout
wait
; echo exec pwd > ctl
; cat stdio
/home/pravin/projects/inferno/ericvh/ericvh-brasil
; cat status
cmd/0 3 Done /home/pravin/projects/inferno/ericvh//ericvh-brasil/ pwd


You can see in the output above that command was executed by ericvh-brasil, which is remote inferno instance and not by inferno-rtasks.

Now, few comments about implementation.
The read call on /taskfs/remote/clone is modified to first perform a read on /remote/local/clone and save the path of reserved resource in REMOTE_RESOURCE_PATH string.

Also, the write call on /taskfs/remote/n/ctl are captured and then written to /remote/local/r/ctl. Similarly, the read on stdio,stderr and status are pulled from their same remote counterpart at /remote/local/r/.

One major flow in this implementation is the way handle to remote resource is stored. In this implementation, this handle is stored in REMOTE_RESOURCE_PATH string which contain absolute path. With help of namec function from emu/port/chan.c, Channel is created separately with each above mentioned remote files.

Correct way to implement this would be to store the handle to the Channel pointing to the /remote/local/r/, and use it to locate all the files without depending on absolute paths. Next version of taskfs will attempt to do this.

Thursday, January 14, 2010

Pitfall : Executing wrong emu

I have wasted 5 hours today in executing wrong emu image and wondering why my changes are not producing any results. This was such a stupid and unexpected mistake that I never considered it.

I have copied the source code from it's initial compilation location to the new location. I did modified the script which starts the emu to point at correct emu image, and also use correct inferno root fs. But I forgot to change the root path variable in mkconfig. All my changes were affecting emu image in old location, while I was running emu image in new location for testing purpose.

So, whenever you copy inferno source-code base, remember to adjust following paths.

  1. The root path variable in mkconfig file. This is absolute path, which means you have to edit it.

  2. $PATH variable that you modify to include Linux/386/bin so that compilation tools like mk can be found.

  3. For running, you need path to the Linux/bin/emu generated after compilation.

  4. For running, you also need a path to the inferno_root filesystem as command line parameter to emu

Wednesday, January 13, 2010

Remote mounting and using devcmd2 filesystem

This blog documents how to mount and use devcmd2 filesystem remotely.

For this, first step is to bind and export devcmd2. In my case, devcmd2 is automatically mounted at /cmd2 location. You can read previous blog for how to do this manually. Now for exporting this filesystem, use following command.

styxlisten -A tcp!*!6666 export /cmd2

And you are done!!!

Any inferno system that wants to mount this filesystem can do it by following command.

mount -A tcp!127.0.0.1!6666 /mpoint

This command mounts the filesystem exported at address tcp!122.0.0.1!6666 on location /mpoint. -A disables any authentication as we are not using any.

Now lets execute commands on remote inferno filesystem.

; cd mpoint
; ls
local
; cd local
; ls
arch clone env fs ns status
; cat clone
0
; cd 0
; ls
args ctl env ns status stderr stdin stdout wait
; echo exec hostname > ctl
; cat stdio
BlackPearl
;

Wow!!! We just executed remote command using Inferno.

This mounted filesystem can be unmounted by unmount command.

unmount /mpoint



Next step is to mount this inferno devcmd2 from Linux. For this you will need to install 9mount, which can be done as follows for debian/Ubuntu based systems.

sudo apt-get install 9mount

And, now use following command to mount it.

9mount 'tcp!127.0.0.1!6666' mpoint

You can unmount this filesystem by
9umount mpoint
command. But before unmounting it, lets try and execute some remote commands using this mount point.

$ cd mpoint
$ cd local
$ ls
arch clone env fs ns status
$ cat clone
0
$ cd 0
$ ls
args ctl env ns status stderr stdin stdio stdout wait
$ echo exec hostname > ctl
bash: ctl: Unknown error 526
$ cat stdio
$ cat stderr
$ cat status
cmd/0 0 Closed /usr/share/lxr/source/inferno//ericvh-brasil/ ''

Well, this is bad news as I am not able to execute commands on inferno from Linux. I also tested with devcmd filesystem, but it gave same error.

bash: ctl: Unknown error 526


Somehow, ctl is not willing to accept commands when sent from sent from mount point on Linux. I don't know the reason behind it, but it seems some more work is needed on 9mount.

Executing host commands from Inferno using cmd filesystem interface ( '#C' )

This blog is about using ability of inferno to execute commands on host OS using filesystem interface. As this project plans to use Inferno as middleware for distributing jobs transparently this interface is important.

The easiest way to execute host command on inferno is by using os command.

; os ls /

Above is the example of executing ls / on host OS.

For using filesystem interface, you can use devcmd or devcmd2 synthetic filesystems.
devcmd2 is extension of devcmd with few improvements. We will stick to devcmd
for purpose of this blog.

You can go into this filesystem by either doing cd or by binding it.

; cd '#C'

This filesystem can be binded to your namespace by following command.

bind '#C' /cmd
cd /cmd


Now, lets see how we can execute commands using this filesystm.

; cd cmd
; ls
clone
; cat clone
0
; cd 0
; ls
ctl
data
status
stderr
wait
; echo exec fortune > ctl
; cat data
Hope that the day after you die is a nice day.
;

What has happened above is we created new resource by catting clone. This resource was 0. So, we cd to 0 for using this resource. We execute the fortune command by echoing it into ctl file. You can execute any command here. After this you can cat the data file for command output and stderr for any errors.

For devcmd2 output will be in stdio file instead of data.

If you don't have fortune command on your system, then go ahead and install it using apt-get install fortune. It is brilliant way to waste time :-).

Tuesday, January 12, 2010

Setting up lxr for Inferno

Understanding and studying the code is much easier if you have source code cross reference set up. There are public repositories for most well known operating systems. You can find them at http://fxr.watson.org/.

Unfortunately, Inferno source code is not hosted here. So, I had to go through the endeavor myself. There are few tutorial out there which gives simple steps for installing and configuring lxr for Linux source code. here are few examples that I referred.
http://tread.wordpress.com/2007/08/25/howto-setup-lxr-on-linux-ubuntu-feisty/
http://orangespike.ca/content/ubuntu-710-lxr-linux-cross-referencing-howto

But I could not find anything which tells how to do it for generic code. So, here I will describe the steps I took for configuring generic code in lxr. Inferno code is taken as example here. It should be easy to replace Inferno with other system code in following steps.


sudo apt-get install apache2 libapache2-mod-perl2
sudo apt-get install libcgi-simple-perl libcgi-ajax-perl libhtml-parser-perl libtemplate-perl


Install LXR.

sudo apt-get install lxr


If possible, install glimpse also. It provides free-text search capabilities. apt-get did not worked for this one in my case. And I did not fight to get it installed also. If following command does not work for you then you can try to get it from http://webglimpse.net/.

sudo apt-get install glimpse


Add the following lines to /etc/apache2/httpd.conf at the end:

# Linux Cross Reference Stuff
Alias /lxr /usr/share/lxr

Options All
AllowOverride All


Create a file /usr/share/lxr/http/.htaccess which contains:



SetHandler cgi-script



Restart Apache

/etc/init.d/apache2 restart



Create /usr/share/lxr/source if it doesn’t exist.

sudo mkdir -p /usr/share/lxr/source
cd /usr/share/lxr/source
sudo mkdir inferno
hg clone https://ericvh-brasil.googlecode.com/hg/ ericvh-brasil
echo "inferno" > /usr/share/lxr/source/versions
sudo ln -s /usr/share/lxr/source/inferno /usr/share/lxr/source/defversion


I performed following steps to get ownership of these file, so that I can edit them directly. You may not have to do these steps. Just make sure that the code directory is accessible to everyone for read and browse.


cd /usr/share/lxr/source
chown -R pravin.pravin inferno
chmod -R +x inferno
chmod 777 /usr/share/lxr/source/inferno/fileidx


Now comes the most important step. You need to modify /usr/share/lxr/http/lxr.conf
to suit your source-code. This file is set-up for linux source-code organization, so, does not work out of the blue for other source-codes. Here is the link to my lxr.conf file. Lines which are modified are highlighted. Most important is to change srcrootname to the root directory of your code. In my case, this entry was modified to

srcrootname: ericvh-brasil


Once you configure /usr/share/lxr/http/lxr.conf, perform following steps.

cd /usr/share/lxr/source/inferno
genxref

Now, your lxr is ready to browse.
Visit http://localhost/lxr/http/source/ericvh-brasil/ to start browsing the code.

If you have system other than Linux or inferno, just try and replace inferno and ericvh-brasil with your version name and source-code root directory and hopefully it will work.

I am quite sure that, this is not the most optimal way to get lxr working. Few of the above steps can be avoided. But, above steps worked for me. Lets hope it helps others as well.

Friday, January 8, 2010

Adding new filesystem to Inferno

I have uploaded the codebase that I will be using on google-code at
http://code.google.com/p/inferno-rtasks/.
Feel free to download and play with it.

My next objective is to add new synthetic file-system similar to /net or /prog to Inferno. Here is the approach that helped me.


First step was to find out where are most of the synthetic fie-systems are residing.
The location was emu/port/. New synthetic filesystem was created by copying
emu/port/devcmd.c with new name emu/port/devtask.c. This file has
one structure Dev cmddevtab at bottom of the file which needs modifications.
I modified this structure as follows

Dev cmddevtab = {
'C',
"cmd",

cmdinit,

Above structure was modified as

Dev taskdevtab = {
'T',
"task",

cmdinit,


You also need to tell compilation process for including this file into compilation path.
This can be done by modifying emu/Linux/emu file which is configuration file
controlling what all should be included in emu image. This file has sections for
dev, lib, link, mod, port, code, init and root. As my addition is supposed to be
synthetic filesystem, I added following line to the dev section of this file.

task


With this modification, you can recompile the kernel and execute the new image.
To test if this new filesystem is present, you can use following command

; ls '#T'
#T/cmd

The output #T/cmd proves that this file-system is in kernel.


Now, lets concentrate on auto mounting this synthetic filesystem at boot time.
This involves few more steps.

First, we need place to mount this filesystem. Lets edit emu/Linux/emu
again. This time, we add following line in root section.

/task /

Refer other entries for the syntax expected in this field. This will create a
placeholder directory /task were we can mount out filesystem.

Next step is to modify initialization script so that it will actually mount synthetic filesystem.
The initialization script can be found at appl/cmd/emuinit.b.
Locate the init() function which is responsible for initial binding of the
filesystem with their mount-points. Add following line to this function after
other filesystems are mounted using similar calls.

sys-bind ("#T", "/task", sys->MBEFORE) ;


I repeated the same modification in appl/cmd/brasil.b as in few cases brasil.b is used
as initialization script.

After this, you should be able to recompile the kernel and test using following command
within inferno.

; ls /task
/task/cmd

Here, the output /task/cmd shows that our filesystem is properly attached to the
root filesystem :-)

Making Host Inferno Usable

I will be working with stripped down version of Inferno for my work.
Code for this model can be accessed at http://code.google.com/r/ericvh-brasil/.

The compilation of this code base involves all steps performed in previous entry.
But in addition to that, one need to create directory
Linux/386/lib before compilation. Otherwise you will get errors in compilation.


Once Installed, I made following changes to make it more usable.

Firstly, following shell-script with name runInferno.sh is used to start the inferno

#!/bin/bash
BASEDIR=/home/pravin/projects/inferno/ericvh/
INSTALL_DIR=ericvh-brasil
$BASEDIR/$INSTALL_DIR/Linux/386/bin/emu -r $BASEDIR/$INSTALL_DIR/


If you execute this script, it will start inferno, which gives ; as prompt.

$ ./runInferno.sh
; wm/wm


The wm/wm command starts the Inferno window manager. If you are not happy with the
size of window manager then you can add -g700x700 parameter in runInferno.sh
shell-script in last line as follows.

#!/bin/bash
BASEDIR=/home/pravin/projects/inferno/ericvh/
INSTALL_DIR=ericvh-brasil
$BASEDIR/$INSTALL_DIR/Linux/386/bin/emu -r $BASEDIR/$INSTALL_DIR/ -g1250x700

1250x700 configuration suits my screen. Feel free to play with these values to
suit your screen-size.

Once window manager starts, there is an icon in left-bottom corner of the window.
You can click on that to get a menu, and you can start shell by selecting the
shell menu option from it.

Here is the final screen-shot of how it looks when window-manager is running.

Thursday, January 7, 2010

Compiling Inferno

Compiling Inferno is little more complicated than installation process.
Web-page for downloading code is here.

Here are the steps I followed.

wget http://www.vitanuova.com/dist/4e/inferno-20091215.tgz
gunzip inferno-20091215.tgz
tar -xvf inferno-20091215.tar
cd inferno


Edit mkconfig file to suit your requirements.
Here is my file

#
# Set the following 4 variables. The host system is the system where
# the software will be built; the target system is where it will run.
# They are almost always the same.

# On Nt systems, the ROOT path MUST be of the form `drive:/path'
ROOT=/home/pravin/projects/inferno/compile/inferno/

#
# Specify the flavour of Tk (std for standard builds)
#
TKSTYLE=std

#
# Except for building kernels, SYSTARG must always be the same as SYSHOST
#
SYSHOST=Linux # build system OS type (Hp, Inferno, Irix, Linux, MacOSX, Nt, Plan9, Solaris)
SYSTARG=$SYSHOST # target system OS type (Hp, Inferno, Irix, Linux, Nt, Plan9, Solaris)

#
# specify the architecture of the target system - Plan 9 imports it from the
# environment; for other systems it is usually just hard-coded
#
#OBJTYPE=386 # target system object type (eg, 386, arm, mips, power, s800, sparc)
OBJTYPE=386

#
# no changes required beyond this point
#
OBJDIR=$SYSTARG/$OBJTYPE

<$ROOT/mkfiles/mkhost-$SYSHOST # variables appropriate for host system
<$ROOT/mkfiles/mkfile-$SYSTARG-$OBJTYPE # variables used to build target object type


Make sure that the ROOT points to the directory created as output of previous tar command.
You also need to add inferno/Linux/386/bin/ path to your $PATH variable.

Once this setup is done then you can compile it by going in emu directory and running mk.

export PATH=$PATH:`pwd`/Linux/386/bin
mk install


In my case, it gave lot of errors :-(
Following is the output of above command till fist few errors.

[emu]$ mk
(cd Linux; mk all)
for i in emu
do
mk 'CONF='$i default
done
cc -c -m32 asm-386.S
cc -c -m32 -DROOT="/home/pravin/projects/inferno/compile/inferno/" -DEMU -I. -I../port -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/home/pravin/projects/inferno/compile/inferno//libinterp -g -O -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/usr/X11R6/include -DLINUX_386 os.c
cc -c -m32 -DROOT="/home/pravin/projects/inferno/compile/inferno/" -DEMU -I. -I../port -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/home/pravin/projects/inferno/compile/inferno//libinterp -g -O -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/usr/X11R6/include -DLINUX_386 segflush-386.c
/bin/sh ../port/mkroot emu
cc -c -m32 -DROOT="/home/pravin/projects/inferno/compile/inferno/" -DEMU -I. -I../port -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/home/pravin/projects/inferno/compile/inferno//libinterp -g -O -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/usr/X11R6/include -DLINUX_386 -I. ../port/lock.c
cc -c -m32 -DROOT="/home/pravin/projects/inferno/compile/inferno/" -DEMU -I. -I../port -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/home/pravin/projects/inferno/compile/inferno//libinterp -g -O -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/usr/X11R6/include -DLINUX_386 cmd.c
cc -c -m32 -DROOT="/home/pravin/projects/inferno/compile/inferno/" -DEMU -I. -I../port -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/home/pravin/projects/inferno/compile/inferno//libinterp -g -O -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/usr/X11R6/include -DLINUX_386 -I. ../port/devmnt.c
cc -c -m32 -DROOT="/home/pravin/projects/inferno/compile/inferno/" -DEMU -I. -I../port -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/home/pravin/projects/inferno/compile/inferno//libinterp -g -O -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/usr/X11R6/include -DLINUX_386 -I. ../port/ipif-posix.c
cc -c -m32 -DROOT="/home/pravin/projects/inferno/compile/inferno/" -DEMU -I. -I../port -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/home/pravin/projects/inferno/compile/inferno//libinterp -g -O -I/home/pravin/projects/inferno/compile/inferno//Linux/386/include -I/home/pravin/projects/inferno/compile/inferno//include -I/usr/X11R6/include -DLINUX_386 -I. ../port/win-x11a.c
../port/win-x11a.c:36:22: error: X11/Xlib.h: No such file or directory
../port/win-x11a.c:37:23: error: X11/Xatom.h: No such file or directory
../port/win-x11a.c:38:23: error: X11/Xutil.h: No such file or directory
../port/win-x11a.c:39:24: error: X11/keysym.h: No such file or directory
../port/win-x11a.c:40:33: error: X11/extensions/XShm.h: No such file or directory
In file included from ../port/win-x11a.c:42:


On googling little-bit, It seemed that some package is missing.
I installed following packages, and it compiled for me.

sudo apt-get install libx11-dev libxmu-dev libxmu-headers

This is not exact list. All three may not be needed. But with these three installed, compilation does work fine :-).

Installing Inferno

For all those who don't know what exactly Inferno is,
please visit wiki page of inferno here
The web page maintained by company is
http://www.vitanuova.com/inferno/


Installing inferno proved to be quite easy task. I have 32 bit Ubuntu-9.10 distribution running, and following are the steps which worked for me. You need to visit Inferno 4th Edition page for downloads and instructions.

Download the CD Image
, then unzip it and mount it using local loopback. Also download the install.pdf which has all the detailed instructions
for this installation.


wget http://www.vitanuova.com/dist/4e/20071003/cdinstall.iso.gz
gunzip cdinstall.iso.gz
mkdir mpoint
sudo mount -o loop cdinstall.iso mpoint/
mkdir inferno_root
sh `pwd`/mpoint/install/Linux-386.sh `pwd`/inferno_root/


With above commands, Inferno should get installed.
The simplest way to test inferno emulator is to run it with following command

`pwd`/inferno_root/Linux/386/bin/emu -r `pwd`/inferno_root/


To shorten the things, one can use following shell-script called run_inferno.sh

#!/bin/bash
BASEDIR="/home/pravin/projects/inferno/"
INFERNO_ROOT="inferno_root/"

$BASEDIR/$INFERNO_ROOT/Linux/386/bin/emu -r $BASEDIR/$INFERNO_ROOT/

Add this file into your ~/bin and start inferno shell just by typing run_inferno.sh