Thursday, June 25, 2009

User interface design for mere mortals

Last night I started perusing the book "User interface design for mere mortals" to see if I could learn something.

Throughout the book, several interesting side topics caught my attention. These include a history of UI's, paper prototyping (www.paperprototyping.com), help systems (robohelp), and psychological types (Briggs and Myers).

The book also mentioned the Archy interface, which I found quite interesting. Archy led me to Enso, and Enso led me to Gnome Go. I started using Gnome Go last night, and I can't believe I hadn't heard of it before. The perfect complementary tool for someone who is used to tab completing (and recursive searching) in a bash shell and who has recently started using Spotlight in MacOSX for a similar purpose.

After looking into the Go source, I realize I should really get to know some more libraries (Gconf, dbus) and other technologies (Mono, C-sharp).

Tuesday, June 9, 2009

Domesticated beaver

No, it's not that kind of "beaver".

Today, while riding through the river valley (in Edmonton), I noticed a man crouched down by the side of the river bank, with a beaver standing right beside him. At first I thought maybe it was as stuffed animal or something, but this guy was actually hand feeding this beaver, who was just chillin' on his back legs, munching away.

I stopped and waited for my girlfriend to catch up--and we just looked and watched these two.

There is often a beaver lurking around that part of the river, but I have never seen one so close to people. Eventually we continued on our way, slightly happier after seeing some beaver. On my way back through the river valley (several hours later, after helping my old man paint his house), climbing a hill, I saw a woman standing with her bike on the side of the trail. I thought nothing of it, and continued to pedal by, making sure to avoid eye contact and look in the other direction. And sure enough, in the trees on the river bank, I saw a beaver standing there looking out in the trail (only a couple of feet from the side of it). Surely it was the same beaver as he was maybe 50 feet from where we had seen him earlier. And surely the woman was staring at how unbothered this beaver was by humans, not unlike we were hours before.

I will have to start increasing the outings to this particular part in the valley, as today's events, combined with others (e.g., people walking their pot-belly pig through the trails in the river valley), it seems like a pretty good place to find randomness.

Sunday, May 17, 2009

Multi-grid



I recently learned about multi-grid methods and was trying to use them in an optic flow setting. I wasn't getting the speed-up that was suggested by a paper on the subject, so I figured that perhaps my reduction or prolongation operators were wrong. To test their correctness I decided to try and solve some simpler problems: a 1D Poisson problem; a 2D Poisson problem; and a 2D non-linear poison like problem. In all cases I used a Gauss-Siedel relaxation. In all my tests I simply ran a multi-grid solver (w-cycle) and then used only the relaxation (Gauss-Siedel) to solve the problem to the same residual.

For the linear problems (Poisson) I am not convinced that MG is the fastest approach. But in the 1D case, it was possible to get speed-ups of 3.3, 14.2, and 38 for problem sizes of 256, 512, and 1024 respectively.

For the 2D linear problem, on sizes of 64x64, 128x128, and 256x256 the speed-ups were 4.8, 14.6, and 51. I have yet to try and solve one of these sparse problems with a generic sparse linear equation solver, but I think the MG solver would be faster. For example, the 256^2 problem took roughly 3.5 seconds, a 512^2 takes about 18.2, and 1024^2 takes roughly 1 minute. Keep in mind the 1024^2 problem is a 1e6 x 1e6 linear system with a million unknowns.

For the nonlinear 2D problem (which is most similar to my target application) the gains were even more apparent. For problems of size 32x32 the Gauss-Siedel method with frozen coefficients had extremely poor convergence. In the same problem size it was possible to see 1000-fold speed-ups. Since the Gauss-Siedel solver alone was extremely slow, I was not able to get results on any larger systems. The non-linear versions also had some strange behaviour, where the residual of the equation system would actually increase after prolonging the coarse grid solution (this would of course decrease after the pre-smoothing for the next iteration).

Back to the optic flow problem. Confident after seeing the improvements in these toy examples, I went back to the optic flow problem. I think that the prolongation and restriction operators in this case are implemented correctly. I didn't see the speed-ups mentioned in the literature (close to 500 or so vs. a Gauss-Siedel solver); mine were closer to 15 or so. The good news is that a single wcycle gives almost real-time performance (as in the paper), where it is possible to get roughly 1 frame-per-second on 400x300 images.

Wednesday, April 22, 2009

get_video, keepvid, and youtube videos

keepvid doesn't seem to be working anymore for grabbing videos from youtube. In the past, I needed a video from Metacafe and couldn't find a utility for getting the video, and the source in the page didn't reveal any obvious links.

As a reminder to myself, the tcpdump command is somewhat helpful in this situation:

tcpdump -A -i eth0 -s 10000 | grep -A6 -B6 "GET"

Using the above command (I think the -s 10000 is redundant), it is somewhat easy to find the command to grab a video. With you tube videos (be sure to set the fmt= parameter to whatever you want) and redirect your browser to the www.youtube.com/watch?... page). The tcpdump will show you the GET get_video?video_id... that you need to grab your video. I think with the newer formats, you may need to use the videoplayback?id GETs.

Still need to write a script to process that stuff for me.

Tuesday, March 17, 2009

perly pie

Thats right, perl "-p -i -e"!

This command is your friend for replacing strings in a batch of documents at once. As I am not a frequent perl user I tend to forget the order syntax for specifying search/replace, but know the rough format contains a "/s" a "/" and a "/g". I tend to try out a quick command line version trial with stdin before going to the full replace. Today I did the test with

perl -p -i -e 's/goodbye/hello/g'

As it worked first try, I then accidentally used the above command in my "find" expression to do my desired replace (e.g., "find ./ perl -p -i -e 's/hello/goodbye/g' {} \;" ). I totally wasn't thinking, as I really wanted to replace occurences of an old host name in the ".svn/entries" files with a new one. Unfortunately, there were actually occurrences of hello in some binary files (I guess I had blender files with objects called hello). Blender then opened these screwed up files without error but containing no objects (the change in length must have 'effed the offsets in the binary). Luckily these were the only files affected and it was easy enough to change. In the future I'll have to make sure to replace the above command with the appropriate search/replace string and a "-name" to find.

OOps.

wxWidgets documentation

Last week I was porting a wxWidget application to the Mac, which shouldn't be all that painful, but I was having trouble with the application opening command line arguments.

Being one of the first GUI programs that used the command line for loading files, I didn't realize that the command arguments (e.g, "open -a myapp.app anargument", don't get passed in through argv/argc). After doing some digging, I turned up a reference for MacOpenFile in the wxPython port. I looked at the current versions documentation (wx 2.8.x) to see if this was something available in the c++ libraries as well. Didn't appear to be on their page (and doesn't come up if you search the site for MacOpenFile). WTF wxDocs?

Anyway, MacOpenFile does exist but just doesn't appear to be documented (in the common docs that is). It is really too bad because I suspect that most people would check wxApp first for alternatives methods to access command line arguments. Instead, you end up wasting time scouring the web for others with similar problems.

I meant to post this right after I had found the solution so I could tag the blog with all the things I had searched for. Unfortunately, I waited more than a week and no longer remember what I had searched.

Friday, March 6, 2009

float cast underflow slow

Today I noticed that some of code that just blended some images was being really slow dependent on some of the coefficients used in the blending. After checking the coefficients for strange values (nans/infs), for which there were none, it was still slow. The problem was only appearing in one function used for blending (some integer code that used floating coefficients that were casted from doubles). The problem didn't appear when strictly using floating point arithmetic. Turns out it is due to slow casting from double to float when there is underflow.

Some timing results for casting from double to double (d), double to float (f), double to integer (i), for test program, included below:

Without optimization
time for d (init=1e-30) 0.013476
time for f (init=1e-30) 0.012164
time for i (init=1e-30) 0.016217
time for d (init=1e-50) 0.019088
time for f (init=1e-50) 0.229608
time for i (init=1e-50) 0.019010

Without optimization (-O3)
time for d (init=1e-30) 0.005358
time for f (init=1e-30) 0.004038
time for i (init=1e-30) 0.003713
time for d (init=1e-50) 0.004566
time for f (init=1e-50) 0.135538
time for i (init=1e-50) 0.003527

You can see the floating point cast is something like 30x slower for the value of 1e-50 (where there is underflow) as opposed to the case of 1e-30 where there is no underflow.

Try the code for yourself.


#!/bin/sh

cat <<EOF > _f_.cc
#include
#include
#include
#include
#include

template
void test(int m, double init){
double * v = new double[m];
T * f = new T[m];
struct timeval tv;
gettimeofday(&tv, 0);

for(int its=0; its< 20; its++){
v[0] = init;
double * vptr = v;
for(int i=0; i< m; i++){
*(vptr++) = v[0];
f[i] = ((T)v[i]);
}
}
struct timeval tva, el;
gettimeofday(&tva, 0);
timersub(&tva, &tv, &el);
printf("time for %s (init=%g) %lf\n", typeid(T).name(), init, double(el.tv_sec) + double(el.tv_usec)/1e6);
delete [] f;
delete [] v;
}

int main(int ac, char * av[]){
int m = 100000;

test(m, 1e-30);
test(m, 1e-30);
test(m, 1e-30);

test(m, 1e-50);
test(m, 1e-50);
test(m, 1e-50);
return 0;
}
EOF

echo "Without optimization"
g++ _f_.cc -lm -o _f_
./_f_

echo "Without optimization (-O)"
g++ _f_.cc -lm -o _f_ -O
./_f_

echo "Without optimization (-O3)"
g++ _f_.cc -lm -o _f_ -O3
./_f_

echo "Without optimization (-O3)"
g++ _f_.cc -lm -o _f_ -O3 -mmmx -msse
./_f_

rm _f_.cc _f_

Tuesday, February 3, 2009

backtrace and friends

Just some notes on instrumenting code using gcc.

Add hooks before and after function calls with "-finstrument-functions" (__cyg_profile_func_exit, ...).

Several ways to look up symbols:

backtrace, backtrace_symbols

libbfd (library used by nm)

create your own symbol table (have macro to map function pointer to name).


On mac, atos is useful.

More to come.

Wednesday, January 28, 2009

WTF apple!

Just spent a long time (as in several hours) stumbling on things that shouldn't have even been problems.

1) Multi-touch on the iPhone. The freaking canvas has to have a property set to enable multi-touch. You still get multi-touch if you don't enable it, but the data is random--which makes you think that the scale and translation you are applying to your canvas is wrong. I see no good reason for having the option of turning off multi-touch events, so whoever decided that I want my wasted hour back.

2) OpenGL errors on the iPhone. Put in some glGetErrors to trace down the few errors I was having. Oddly enough they seemed to be right around the calls that used GL_CLIP_PLANE1 (not GL_CLIP_PLANE0, just clip plane 1). Sure, that didn't seem to be a problem before, but what'evs--I can deal with it. Still some other textures that looked wrong. Turns out I inadvertently created some non-power-of-two textures (which work fine in new versions of OpenGL now-a-days). As soon as I realized this I decided to double check the clip plane problem. Sure enough, no longer a problem. Why in the hell was an error being reported on the CLIP_PLANE1 and not somewhere closer to the texture. Damn, damn, double damn.

Sunday, January 18, 2009

And again



I found live ammunition on a bridge late last night. Made my day.

Still not much more luck with the lines. This rasterizes the current line, computes the signed distance from the line, extracts the level set at a specific distance, and then tries to connect the resulting polygons to the existing line. That's why the line appears to follow itself.

Slow. Downsampled they almost look like the embedded code images.



Wednesday, January 14, 2009

Line update



Some more lines. I have tried several different methods, and still nothing is giving me what I expect.


The code (which is a mess) because I wrote it at 3 in the morning is also embedded in the other not-so nice image (as usual).

Static is good, and static is great, but static initialization order sucks. I have had problems that could be attributed to static initialization in c++ before, but I wasn't entirely certain at the time that this was the problem. Just like an ugly memory bug, the static initialization order rears its head with meaningless memory access violations--which in my case seem to always end up deep in the stl. While porting something from linux to Mac, a simple re-ordering of the object files on the make line was enough to solve the problem. Wrapping the static variables in static functions to ensure an initialization order is clearly a better solution.

Obviously these things should become a problem and cause problems before main is even entered. I was also building the application for the iPhone and was experiencing problems with it as well. The problem was a segfault of sorts, and the stack was getting f'ed, and GDB was being almost hopeless. Of course, as I just had problems with static initialization I attributed these problems to be that beast. After several failed attempts at re-organizing the static initialization, digging through the semi-automatic resource handling code--I eventually followed the trail down the wormhole with the debugger to a buffer underflow error. Damn it! The only part of code that was different between the Linux/Mac version and the iPhone version was the loading of resources--which first have to dereferencee file names from the main application bundle. This code was different, and in the iPhone port I was using CFString and its reverse search to find the last occurrence of the '.' for file extensions. Obviously I wasn't checking to see if the return value was within bounds before computing a substring--which suprisingly wasn't being checked in the CFString substring functions. I'm certain I spent an hour looking for that. So obvious, but I was led astray by the static initialization. Cut again by Occam's---again and again and again.

Thursday, January 8, 2009

Arduino



Got an arduino for x-mas and just got around to playing with it the last couple nights. Pretty damn fun. Learnt a little (pull-down and pull-up circuits).

Also started working on generating an image from a continuous line. Preliminary results and some embedded source code are linked in this page. The version is based on subdivision and post-processing.