Friday, December 24, 2010

Getting to the root of the gnuplot troubles

As I mentioned in my previous post, the gnuplot in Ubuntu does not work with Lush and that you have to build it yourself. The make-or-break difference between Ubuntu's and your own built is that Ubuntu's gnuplot links libeditline whereas your own built should link the GNU readline library. That's how far I got last time I looked at it, but I did not dig deeper to understand what gnuplot with libeditline does differently from gnuplot with libreadline.

The reason that the specific command-line library could have anything to do with the problem I am seeing with Ubuntu's gnuplot is that, unlike any other gnuplot interface I have seen, Lush's interface is bidirectional. The nice thing about it is that you can create a Gnuplot instance and use it to interact with gnuplot from within the Lush shell.

? (libload "gnuplot/gnuplot")
[gnuplot.lsh]
[gnuplot-config.lsh]
= "/home/juenglin/downloads/lush2-ref/packages/gnuplot/gnuplot.lsh"
? (defvar gp (new Gnuplot))
= gp
? (gp "show version")

G N U P L O T
Version 4.4 patchlevel 2
last modified Wed Sep 22 12:10:34 PDT 2010
System: Linux 2.6.32-26-generic

Copyright (C) 1986-1993, 1998, 2004, 2007-2010
Thomas Williams, Colin Kelley and many others

gnuplot home: http://www.gnuplot.info
faq, bugs, etc: type "help seeking-assistance"
immediate help: type "help"
plot window: hit 'h'

= ()
? (gp "show terminal")

terminal type is wxt 0

= ()

A bidirectional interface is more complicated to implement than a unidirectional interface. The nuts and bolts of a unidirectional interface consist of creating a pipe, spawning a gnuplot process and making the read end of the pipe the processe's standard input. Use the write end of the pipe to send commands and data to gnuplot and you are basically done.

What makes a bidirectional interface more complicated is not so much that one needs to use two pipes, but the difficulty of staying in sync with gnuplot without getting into a deadlock. The approach of the current implementation of Lush's bidirectional interface is to launch gnuplot in interactive mode and to use the gnuplot prompt as a cue for when to stop reading from the receiving pipe. The protocol basically mimics the interaction a user has within the gnuplot shell: enter a command, type return, see gnuplot's answer and a new gnuplot prompt.

Now, what's wrong with libeditline? It turns out that when running gnuplot with libeditline in this mode, gnuplot never sends a prompt. I haven't quite figured out why that is the case. But as an experiment I connected to gnuplot via a pseudo terminal and it turns out that gnuplot with libeditline then sends the prompt and the interface works. Unfortunately, using pseudo terminals for the interface is not a solution as the communication seems to go a lot slower than with pipes.

Wednesday, December 22, 2010

Using Lush with Gnuplot

I had not been using Lush for about a year and yesterday I wanted to write a few lines of code to recreate a plot from a linear algebra textbook. This is the sort of thing that people turn to Matlab for: write a few lines of code to generate some data, call plot once or twice, done. When I am on the top of my game I can do the same with Lush, it would not take me more than ten minutes.


The plot in question shows the roots of Wilkinson's polynomial in the complex plane and illustrates that polynomial root finding is an ill-conditioned problem. The thicker black dots are the known roots of the polynomial and the many little red dots are the roots returned by a polynomial root finder after perturbing the polynomial's coefficients slightly. More precisely, the program randomly perturbed the coefficients, called the root finder and plotted the roots in red. Repeating this 100 times you get a plot like the one above.

The point of the plot is that the exact location of the roots as a function of the polynomial coefficients is very sensitive for this polynomial. Perturbing the coefficients cause changes in the roots' locations many orders of magnitude larger than the changes in the coefficients.

The point of this blog post, on the other hand, is to show how easy it is to generate that plot in Lush. Here is the function that created the plot (the complete script is in Lush's demos directory).

(defun wilkinson-demo (&optional (n-pertubations 100))
(let* ((p *p-wilkinson*)
(n ($n *p-wilkinson*))
(s ($> [@ 1e-10] n-pertubations n))
(coeffs (* ($< p n-pertubations) (+ 1 (rand s))))
(plotter (new Gnuplot 'interactive ())) )
(plot
(title "Roots of Wilkinson's polynomial")
(points (++> *roots-wilkinson* [0]) (lc 'black) (pt 7) (ps 1))
(do ((c coeffs))
(points (poly-solve c) (lc 'red) (pt 7) (ps 0.2)) ))
plotter))

Just a few lines, nice. But yesterday it took me more than half a day to get this done. Not because I forgot so many things about the language, but because I had not fully rebuilt everything I needed after upgrading my laptop a few months back: The gnuplot installed on Ubuntu simply did not work with Lush and I had to figure out/remember why.

Long story short, if you want to use gnuplot with Lush (you do), then you need to build it yourself. That is, download the sources, install all the development packages for the dependencies that make gnuplot most powerful and pleasant to use (libreadline, wxWidgets, cairo, pango, libgd), and build it.

I installed the most recent version 4.4. and the gnuplot demos in Lush all worked fine.