Syntax Highlighting

Monday, 7 January 2013

Passing Functions in D

So, the problem is to create a function (foo) that:
  • takes two values xand x2.
  • returns a a function that 
    • takes as argument a single number
    • return the  interpolated value between the two intial values
For example:
x1 = 10
x2 = 15
f(x1, x2) = (x2 - x1)*t + x1 = F(t)
F(0.5) = 12.5


import std.stdio;

auto interp_proc_maker(float x1, float x2){
    float interp(float s) {
        return (x2-x1)*s+x1;}
    return  &interp;}

void main()
{
  auto f1 = interp_proc_maker(10, 15);
  writefln("f1(0.5) = %f", f1(0.5));
}

$ gdc example.d
$ ./a.out
f1(0.5) = 12.500000

Saturday, 5 January 2013

Call Graph in Xfoil

Use the following procedure to visualise the logic through xfoil:
  1. Make sure you have the correct dependencies
    1. gprof (or one of the alternative profilers installed).
    2. Make sure you have graphviz 
    3. Download gprof2dot.py
    4. Make sure that you can compile xfoil properly
  2. Modify the flags in the Makefile (${srcdir}/Xfoil/bin/Makefile) to allow the profiling of the xfoil code.
    1. Add the -pg option to the FFLAGS and FFLOPT makefile variables.
    2. Add the -pg option to the make xfoil command line ("    xfoil: $(XFOILOBJ) $(XUTILOBJ)... $(FC) -o xfoil $(XFOILOBJ) .... -pg"
    3. Note it may be necessary to do a "make clean" to be able to recompile xfoil.
  3. Now create a script file that touches all the xfoil commands that your interested in. 
    1. Example below does an viscous, (slightly) compressible analysis analysis of a NACA 23012 airfoil section at 10 degrees AoA.
  4. Now run the script.
    1. ~/bin/xfoil < naca23012.xfoilscript
    2. Note that I have a bin directory in my home folder (~) that has a symbolic link to the particular xfoil program I want to run. Hence ~/bin/xfoil command.
  5. If you've compiled xfoil correctly you'll get profiler file (gmon.out for gprof) file in the same directory that you ran the xfoil script in.
  6. Now run the gprof2dot script to generate a dot file.
    1. gprof ~/bin/xfoil | ./gprof2dot.py -n0 -e0 | dot -Tpng -o xfoil.png
  7. Enjoy the call graph
Xfoil call graph for the example script



Xfoil Script


$ cat naca23012.xfoilscript
plop
g


naca
23012

oper
v
3e5
m
0.3
a
10


quit

Tuesday, 1 January 2013

Designing a System for Technical Skills Progression

Found in drafts file, Started Jan-2011

First some background about me: I work two part-time jobs, Mondays and Tuesdays I'm an engineer, Wednesday through to Friday I'm a draftee, although at the moment I'm doing some excel/multiframe programming/engineering analysis design.

The situation at my engineering company is this: foreign company required by government contract to provide certain percentage of work in country. That's us. Effectively our client is the government, and we represent the foreign company. Most of our (maintenance, very important point) work uses a streamlined process, where there is the minimum legal amount of oversight on the work that we provide in order to improve the overall efficiency / effectiveness/ turn-around time; and that is the work that I have been mostly doing for the past year or so. However, recently I was tasked with a modification instead of maintenance.

My manager decided to step with me through how our company handles on-going multi-week projects (most of my projects are expected to be completed within a week). I estimated about 60 hours to complete (about 4 weeks). He laughed. After going through the project times with admin, checking, writing, drafting, etc. I came up with 144 hours (about 10 weeks) which includes a 10% risk (I like to think of this as a training allowance). Now compared with my previous 15 hour project turn-around time I felt this was hideously over budgeted. As it was, the project ended up around the 135 hour mark, smack bang in the middle of the risk margin.

So how was I so very wrong in my estimation for the budgeting?

My first 'proper' job was doing similar kind of engineering but for the commercial sector. The turn-around times and budgets are much tighter than for government work. Also, access to OEM data is restricted. As a result, a higher level of engineering risk is accepted and use of engineering judgement was favoured (and often was the only engineering justification). My boss at the time was an engineer of about 35 years experience, running his own business (there were only two of us; I think the peak employment for his company was four employees). On occasion, I would not be able to understand why a particular solution was acceptable, to which he would reply that "based on my experience and knowledge - it will be okay".

This annoyed me.

I rebelled by making forms. And checklists. And cheat sheets. I even started on a manual. Of course I was trying to systematise the process for the engineering we were doing; trying to capture the skills and abilities of my boss, by without having to go through the time taken to gain that experience. This strategy is not uncommon. Although there was resistance to this, some forms later proved useful in the form of third-party audits. One thing I was not able to solve was the "don't know what you don't know" problem.

As an aside, it seems to me that the solution the pro-systems camp adopt is one of oversight, review and approval. However, at a proto-level, oversight, review and approval is still being performed by people; and so you are still vulnerable to the "you don't know what you don't know" problem. The systems' solution is, of course, to add oversight, review and approval to the people doing the overseeing, reviewing and approving...

Adding these meta-levels of process to a problem at the proto-level can only ever dilute the problem, not eliminate it.

The authority (e.g. Civil Aviation Safety Autority) has the problem of how much authority to delegate, to little and the delegation has no purpose. Too much and delegatee is vulnerable to the Dunning-Kruger effect. Myself, I go for the personal education and training as a defense to this.

Now my draftee work has had problems with their quality. Jobs have gone out that are straight out unfixable. This has meant massive back charge to us. Recently, our manager was promoted across the country, and our final checker was promoted into his position. What this has meant is that the new final checkers were now checking work for which they were not skilled enough to do. In addition the only training is on-the-job training. The person most recently able to be productive is tasked with the training new hires (an effort to solidify the training of the productive person). Any problems or questions are forwarded to one of the more senior draftees for resolution. What this has bought about is draftees who have been taught to obey the instructions of the engineers and engineers who have not been trained to an adequate level.

Another symptom is the sudden push at the end of every month to get jobs out. Ideally, there should be no push for jobs out as the budget should be comfortably passed near the end of the month with no change of pace. But there is: so what does that mean?

We now have three situations in mind. The first is my preference to for systems when first starting out. The second is a poor estimate of project time budget. And the third is the significant drop in quality of the drafting work. To provide a paradigm on which to analyse this I adopt the Dreyfus Model of Skills Aquisition. This model sets out 5 levels at which a person's skill can be categorised. Quoting Wikipedia:
  1. Novice:
    • "rigid adherence to taught rules or plans"
    • no exercise of "discretionary judgment"
  2. Advanced beginner
    • limited "situational perception"
    • all aspects of work treated separately with equal importance
  3. Competent
    • "coping with crowdedness" (multiple activities, accumulation of information)
    • some perception of actions in relation to goals
    • deliberate planning
    • formulates routines
  4. Proficient
    • holistic view of situation
    • prioritizes importance of aspects
    • "perceives deviations from the normal pattern"
    • employs maxims for guidance, with meanings that adapt to the situation at hand
  5. Expert
    • transcends reliance on rules, guidelines, and maxims
    • "intuitive grasp of situations based on deep, tacit understanding"
    • has "vision of what is possible"
    • uses "analytical approaches" in new situations or in case of problem
Now when I first started, by predilection for systems was for precipitated by being a novice in an environment of an expert. In retrospect, this seems reasonable. Effectively, the forms and checklists are being used as training wheels.

In the second situation, I feel I am about the level of advanced beginner / competent. For maintenance projects. What happened was that I based initial estimate.

So what does all this have to with the title? I propose that the system for technical jobs be flexible enough to accommodate the people who are working on it. In the specific example of the my modification project, if I had recognised that my level of skills was closer to novice than to competent, I spend some time with the manager to create some forms and checklists to go through job.

On checklists: The effectiveness of a checklist is equal to the comprehensiveness of the checklist multiplied by the probability that it is used. There is a tendency at my drafting job to fix problems with jobs by adding more items to the checklist. What this results in is a checklist that is 80% not applicable. However, hidden in amongst the "n/a" is a checklist item required that gets overlooked. Or you 'check' the same way that you do create, in which case a mistake is 'checked' OK. You can't check your own work, especially if you've just finished. I'm always surprised at the number of mistakes that sneak in overnight.

TLDR; Customise checklists based on the job and the person doing it.

ProgEx: Python Ex1 rev 1 comments

  1. The code won't run from the command line under linux, nor does it run from the Python command line. Feature request?
  2. I made a change to how the sqrt function works (it was even more 'hacky'). Actually the whole check for negative numbers the using the complex math library seems a bit clunky. Options? Major sin: I didn't check to see if it works. I'll post it anyway.*
  3. On that note, I have a feeling that the sqrt function only returns the first value. Fails functional spec. D'oh.A comment on the spec: three of the functions expect a single return value, however the fourth (sqrt), expects a list to be returned. Something that should have considered in the 'architecture'.
  4. Should the doit() have been called main(), and the output in that routine, instead of printing the result of doit()?
  5. The shebang line to make it run on Linux was expected. It was a bit of easter egg to try and get people to check further...
  6. I hadn't got up to the chapter on list comprehensions, or I've skipped it.
  7. sys.exit instead of exit. That appears to be a bit of a trap.
  8. It should have unit testing in the module specified. 
  9. Docstrings are absent.

ProgEx: Python Ex rev 1 Code


# Number functions

# A program written for Exercise 1 of 15 programming exercises.
# As specified in knowing.net
# The aim of this exercise is to familiarise with basic code constructs,
# basic operators, standard libraries, simple maths, arrays, functional
# programming concepts (map and apply?), parsing input, error handling, etc.
#
# The aim of this module is to create a simple program that will
# take as its first argument one of the following arguments:
# sum, product, mean, sqrt,
# The remainder of the arguments will be a list numbers that the
# functions apply to.
# This module is inteded to be called from shell of the OS., eg
#
# c:\python\exercises\numberfun.py sum 2 3 4 5
#  14
#

# Extensions and maintenance guide:
# To add extra functions:
#   - Define the function in the Function Definitions
#   - Add the function to the function table with its 'call'
#   - Implement any exceptional behaviour in the main program and
#         document it.

from __future__ import division
import math
import cmath
import sys

# The module is divided into the following sections:
# 1) Function Definitions
# 2) Function Lookup table
# 3) Main program definition
# 4) The command to run

### Function Definitions
## sum
#def sum(lst):
#    return reduce ((lambda x,y: x+y), lst, 0)
#
# sum is an inbuit function

## product
def product(lst):
    return reduce ((lambda x,y: x*y), lst, 1)

## mean
def mean(lst):
    return (reduce ((lambda x,y: x+y), lst, 1)) / len(lst)

## sqrt
def sqrt(lst):
    return map (math.sqrt, lst)

def csqrt(lst):
    return map (cmath.sqrt, lst)

## help
def numberfunhelp():
    print """"
    Help for numberfun: an exercise in learning."""
## Function table:
functiontable = {'sum': sum,
'product': product,
'mean': mean,
'sqrt': sqrt,
'help': numberfunhelp}
#########################################################
### Program Starts Here.

## Error check the arguments
## and run the program
def doit(lst):    # lst is command line argument strings
    numlst = []   # the numeric equivalent of the argument strings.
    for el in lst[2:]:
        try:
            numlst.append(float(el))
        except ValueError:
            print '%s is not a valid number' % el
            exit
    # final chance for exceptional functions:
    if any(map((lambda x: x&lt;0), numlst)):
        functiontable["sqrt"] = csqrt
    if lst[1] in functiontable:
        return map(functiontable[lst[1]], [numlst])[0]
    else:
        print "Function %s not implemented" % lst[1]
        exit

print doit(sys.argv)


15 Steps to Learning a new language

Larry O'Brien has written a very good blog on exercises to learn a new programming language.  As there are a number of languages to learn, I'll start off with something popular (like Java), but not (like LISP).  

Compiling Xfoil on Debian (squeeze)

OK, first off: if you want xfoil to design aerofoils only, then use the xfoil apt package . If, for some reason, you actually want to compile xfoil, then you can follow these instructions.

Instructions

  1. Decide on where the various files for xfoil are going to be stored:
    1. My preferences are to download the file's to the the Downloads directory in my home folder. Extract the zipped files to a 'src' directory in my home folder, and perform all the compilations there. I have a bin folder in my home directory ( ${HOME}/bin/xfoil ) which has a symbolic link to the final compiled binary in src folder ( ${HOME}/src/Xfoil/bin/xfoil ). 
    2. Also decide on if you are going to use double precision or single precision arithmetic. I chose double precision.
  2. Download the source for the xfoil.
  3. Build Orr-Sommerfeld database in ./orrs
    1. cd orrs
    2. pwd
    3. cd src
    4. edit osmap.f
      1. Find the following line in osmap.f (roughly line 100) : DATA OSFILE / '/var/local/codes/orrs/osmap.dat' /
      2. Take the absolute directory string which is generated by the pwd command above, paste it in front of the osmap.dat filename, e.g. DATA OSFILE / '/home/me/src/Xfoil/orrs/osmap.dat' /
      3. This statement will tell SUBROUTINE OSMAP where to find this table data file. 
      4. Find the following line in osmap.f (roughly line 75): REAL*4 RLSP, WLSP, HLSP, & RINCR, WINCR, RL, WL, HL, & A, AR, AW, AH, ARW, ARH, AWH, ARWH
      5. If you choose to use single-precision for the OS data file (should be adequate), leave this line as is. If you wish to do everything in double precision, change the REAL*4 to REAL.
    5. cd ../bin
    6. edit Makefile
      1. Note that there is a lot of redundancy in the make files. For example, in one of the make files the variable FC is assigned about three times. If you get an ifort not found compile error, then you haven't set the last assigned value to your fortran compiler.
      2. Change the compiler flags to match the Fortran compiler on your system. For debian squeeze this is probably fort77 or gfortran. I am using gfortran and have used the setting below:
        FC = gfortran
        FLG = -O -fdefault-real-8
        PLTLIB = -L/usr/X11R6/lib -lX11 # left as is from the intel fortran settings
        FTNLIB = # left as is from the intel fortran settings
      3. Note that the FLG setting -fdefault-real-8 changes the the compiler treats reals. I suspect that this setting overides the change to osmap.f (Item 3dv).
    7. Make the osgen object. Use Makefile_DP for double-precision OS database file.
      1. make osgen
        make osmap.o
        for single precision (check compiler flag settings) OR for double precision:
        make -f Makefile_DP osgen
        make -f Makefile_DP osmap.o
    8. cd .. 
    9. bin/osgen osmaps_ns.lst (creates binary file osmap.dat)
  4. Build the plot library in ./plotlib
    1. cd plotlib
    2. edit Makefile (set compiler flags for your machine)
      1. Comment out the include ./config.make
      2. If you want to use double precision then
        comment out PLTLIB = libPlt.a and uncomment PLTLIB = libPltDP.a
        #PLTLIB = libPlt.a
        PLTLIB = libPltDP.a
        and now set the compiler flags settings for double precision. Uncomment the DP variable and set the appropriate flag. Mine is:
        #DP = -r8 becomes
        DP = -fdefault-real-8
    3. make
  5. Build the programs in ./bin 
    1. cd bin
    2. edit Makefile
      1. Be sure to set up the FC variable and other compiler flags, FFLAGS and FFLOPT (used for xsolve) as appropriate. My settings were:
        FC = gfortran
        FFLAGS = -O3 -fdefault-real-8
        FFLOPT = -O3 -fdefault-real-8
        INSTALLCMD = install -s
        CC = gcc
        CFLAGS = -O -DUNDERSCORE
        Note: some other instructions use the -c flag with gfortran.
      2. make xfoil
        1. At this point, I got some arithmetic overflow errors:
          gfortran -c -O3 -fdefault-real8
          ../src/xoper.f:117.22:

          IINPUT(I) = 2**31
                                  1
          Error: Arithmetic overflow at (1)
        2. This can be fixed by modifying the source code. You need to make two changes, which are easily implemented using the sed command. Thanks to Judson's Notes.:
          cd src/
          sed -i 's/LOGICAL ERROR, LGETFN/LOGICAL ERROR, LGETFN, LERR/g' pplot.f
          sed -i 's/IINPUT(I) = 2\*\*31/IINPUT(I) = HUGE(0)/g' xoper.f
      3. make pplot
      4. make pxplot
    3. If all works out, then you can run xfoil by going to the $BINDIR specified in the make file 

    Sources:

    1. README available in the xfoil source code download (.tar.gz)
    2. Judson's Notes for Compiling Xfoil on Ubuntu 11.04
    3.  Icaro's Blog for HOW TO Compile Xfoil 6.97 in Ubuntu Linux