 
        Image File System Reference Manual
Version 8.2.6 April 5, 2014
Wesley Snyder
2
Acknowledgements
Based on suggestions by Wesley Snyder of North Carolina State University and Joe
Erkes of General Electrics Corporate Research and Development Center, the staff of Communication Unlimited, Inc. designed and coded IFS version 1 during the fall of 1986. Mark
Lanzo designed and implemented most of versions 1-4, although other members of the CUI
staff contributed heavily, including Katie Boone, Mark Yarboro, Gary McCauley, Bennett
Groshong, and Paul Hemler. Katie Boone is responsible for version 5, and Rajeev Ramanath contributed heavily to the creation of Version 6. Version 7 never really got listed
as a separate version. Version 8 incorporated 64 bit computers, and compatibility with
openCV, and was written mostly by Wesley Snyder and Ben Riggan. IFS is a trademark
of Communication Unlimited, Inc. IFS is jointly owned by CUI and North Carolina State
University.
Contents
1 The IFS system
1.0.1 New with version 8.1 . . . . . . . . . . .
1.0.2 New with Version 8 . . . . . . . . . . .
1.0.3 New with Version 6 . . . . . . . . . . .
1.0.4 New with Version 5 . . . . . . . . . . .
1.1 Using IFS . . . . . . . . . . . . . . . . . . . . .
1.1.1 Program Compilation and Linking . . .
1.2 Referencing IFS images . . . . . . . . . . . . .
1.2.1 Naming conventions in IFS . . . . . . .
1.3 Error handling in IFS . . . . . . . . . . . . . .
1.3.1 Image validation in IFS . . . . . . . . .
1.3.2 Header Files . . . . . . . . . . . . . . .
1.3.3 Coordinate systems and array storage in
1.4 IFS FUNCTION LISTING . . . . . . . . . . .
1.4.1 bilininterp . . . . . . . . . . . . . . . . .
1.4.2 ifsalc . . . . . . . . . . . . . . . . . . . .
1.4.3 ifscigp read from complex or real image
1.4.4 ifscigp3d . . . . . . . . . . . . . . . . . .
1.4.5 ifscipp . . . . . . . . . . . . . . . . . . .
1.4.6 ifscipp3d . . . . . . . . . . . . . . . . . .
1.4.7 ifscfgp . . . . . . . . . . . . . . . . . . .
1.4.8 ifscfgp3d . . . . . . . . . . . . . . . . . .
1.4.9 ifscfpp . . . . . . . . . . . . . . . . . . .
1.4.10 ifscfpp3d . . . . . . . . . . . . . . . . .
1.4.11 Ifsclose . . . . . . . . . . . . . . . . . .
1.4.12 ifscreate . . . . . . . . . . . . . . . . . .
1.4.13 ifs2ipl . . . . . . . . . . . . . . . . . . .
1.4.14 ipl2ifs . . . . . . . . . . . . . . . . . . .
1.4.15 ifsCVpin . . . . . . . . . . . . . . . . . .
1.4.16 ifsCVpot . . . . . . . . . . . . . . . . .
3
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
IFS
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
. . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
9
9
10
10
10
11
11
12
13
13
14
18
19
20
21
21
22
23
24
25
26
27
28
29
30
31
33
33
33
34
4
CONTENTS
1.5
1.6
1.7
1.4.17 ifsCVPopup . . . . . . . . . . . . . . . . . . .
1.4.18 IFSDISPLAY . . . . . . . . . . . . . . . . . .
1.4.19 Building ifsCV and IFSDisplay with openCV
1.4.20 ifsdimen . . . . . . . . . . . . . . . . . . . . .
1.4.21 ifsexwin . . . . . . . . . . . . . . . . . . . . .
1.4.22 ifsexwin3d . . . . . . . . . . . . . . . . . . . .
1.4.23 ifsfgp . . . . . . . . . . . . . . . . . . . . . .
1.4.24 ifsfgp3d . . . . . . . . . . . . . . . . . . . . .
1.4.25 ifsfpp3d . . . . . . . . . . . . . . . . . . . . .
1.4.26 ifsfpp . . . . . . . . . . . . . . . . . . . . . .
1.4.27 ifsfree . . . . . . . . . . . . . . . . . . . . . .
1.4.28 ifsGetFN . . . . . . . . . . . . . . . . . . . .
1.4.29 ifsGetImg . . . . . . . . . . . . . . . . . . . .
1.4.30 ifsigp . . . . . . . . . . . . . . . . . . . . . . .
1.4.31 ifsipp . . . . . . . . . . . . . . . . . . . . . .
1.4.32 ifsmkh . . . . . . . . . . . . . . . . . . . . . .
1.4.33 ifsopen . . . . . . . . . . . . . . . . . . . . . .
1.4.34 ifspin . . . . . . . . . . . . . . . . . . . . . .
1.4.35 ifspot . . . . . . . . . . . . . . . . . . . . . .
1.4.36 ifsPrsFN . . . . . . . . . . . . . . . . . . . . .
1.4.37 ifsPutImg . . . . . . . . . . . . . . . . . . . .
1.4.38 ifsRdHdr . . . . . . . . . . . . . . . . . . . .
1.4.39 ifsRdImg . . . . . . . . . . . . . . . . . . . .
1.4.40 ifssiz . . . . . . . . . . . . . . . . . . . . . . .
1.4.41 ifsslice . . . . . . . . . . . . . . . . . . . . . .
1.4.42 ifsversion . . . . . . . . . . . . . . . . . . . .
1.4.43 ifsWrImg . . . . . . . . . . . . . . . . . . . .
IFS Error Codes . . . . . . . . . . . . . . . . . . . .
IFS Data Types . . . . . . . . . . . . . . . . . . . . .
The structure of an IFS image . . . . . . . . . . . .
1.7.1 The image header fields . . . . . . . . . . . .
1.7.2 The dimension sub-header fields . . . . . . .
2 The FLIP library
2.1 Example Program Using
2.2 Matrix Operations . . .
2.2.1 dmatrix . . . . .
2.2.2 matrix . . . . . .
2.2.3 dvector . . . . .
2.2.4 vector . . . . . .
2.2.5 ivector . . . . . .
the FLIP functions
. . . . . . . . . . .
. . . . . . . . . . .
. . . . . . . . . . .
. . . . . . . . . . .
. . . . . . . . . . .
. . . . . . . . . . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
34
36
37
38
39
40
41
42
43
44
45
47
48
49
51
53
54
55
56
57
58
59
60
61
62
63
64
64
65
66
67
70
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
73
73
74
74
75
75
75
75
CONTENTS
2.3
2.4
2.5
2.2.6 free ivector . . . . . . . .
2.2.7 free dvector . . . . . . . .
2.2.8 free vector . . . . . . . . .
2.2.9 transpose . . . . . . . . .
2.2.10 ifsmatmult . . . . . . . .
2.2.11 ifsinverse . . . . . . . . .
2.2.12 jacobi . . . . . . . . . . .
Arithmetic Operations on Images
2.3.1 flabsolute . . . . . . . . .
2.3.2 fladds . . . . . . . . . . .
2.3.3 fladdv . . . . . . . . . . .
2.3.4 flclip . . . . . . . . . . . .
2.3.5 flcp . . . . . . . . . . . .
2.3.6 fldivs . . . . . . . . . . . .
2.3.7 fldivv . . . . . . . . . . .
2.3.8 flexp . . . . . . . . . . . .
2.3.9 flln . . . . . . . . . . . . .
2.3.10 flmults . . . . . . . . . . .
2.3.11 flmultv . . . . . . . . . .
2.3.12 flneg . . . . . . . . . . . .
2.3.13 flnorm . . . . . . . . . . .
2.3.14 flrec . . . . . . . . . . . .
2.3.15 flsq . . . . . . . . . . . . .
2.3.16 flsqrt . . . . . . . . . . . .
2.3.17 flsubs . . . . . . . . . . .
2.3.18 flsubv . . . . . . . . . . .
2.3.19 flthresh . . . . . . . . . .
2.3.20 flthresh2 . . . . . . . . . .
Image Manipulation Functions .
2.4.1 flone border . . . . . . . .
2.4.2 flpad . . . . . . . . . . . .
2.4.3 flplanar . . . . . . . . . .
2.4.4 flrotate . . . . . . . . . .
2.4.5 flshx . . . . . . . . . . . .
2.4.6 flshxy . . . . . . . . . . .
2.4.7 flshy . . . . . . . . . . . .
2.4.8 flzero border . . . . . . .
Derivative Operators . . . . . .
2.5.1 flDoG . . . . . . . . . . .
2.5.2 flGabor . . . . . . . . . .
2.5.3 fldx . . . . . . . . . . . .
5
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
76
76
76
76
76
77
77
77
77
77
78
78
78
78
78
79
79
79
79
79
79
80
80
80
80
80
80
81
81
81
81
81
81
82
82
82
82
83
83
84
85
6
CONTENTS
2.6
2.7
2.5.4 fldx back . . . . . . . . . . . . . . . . . . . . . . .
2.5.5 fldx forw . . . . . . . . . . . . . . . . . . . . . . .
2.5.6 fldxx . . . . . . . . . . . . . . . . . . . . . . . . . .
2.5.7 fldxy . . . . . . . . . . . . . . . . . . . . . . . . . .
2.5.8 fldxz . . . . . . . . . . . . . . . . . . . . . . . . . .
2.5.9 fldy . . . . . . . . . . . . . . . . . . . . . . . . . .
2.5.10 fldy back . . . . . . . . . . . . . . . . . . . . . . .
2.5.11 fldy forw . . . . . . . . . . . . . . . . . . . . . . .
2.5.12 fldyy . . . . . . . . . . . . . . . . . . . . . . . . . .
2.5.13 fldyz . . . . . . . . . . . . . . . . . . . . . . . . . .
2.5.14 fldz . . . . . . . . . . . . . . . . . . . . . . . . . .
2.5.15 fldzz . . . . . . . . . . . . . . . . . . . . . . . . . .
Running the FLIP functions from the command line . . .
2.6.1 flpg1: single input FLIP functions . . . . . . . . .
2.6.2 der1: take a derivative of an image . . . . . . . . .
2.6.3 ifsDoG:Apply a derivative of Gaussian kernel to an
Image Manipulation . . . . . . . . . . . . . . . . . . . . .
2.7.1 ifsadd . . . . . . . . . . . . . . . . . . . . . . . . .
2.7.2 ifsany2any . . . . . . . . . . . . . . . . . . . . . . .
3 Image Processing Subroutines
3.1 Geometric functions . . . . .
3.1.1 chainHull 2D . . . . .
3.1.2 compute curvature . .
3.1.3 InsideTriangle . . . . .
3.1.4 InterpolateTriangle . .
3.1.5 cubic splines . . . . .
3.1.6 Resampling Curves . .
3.1.7 Operations on FIFO’s
3.1.8 MinimumRegion . . .
3.1.9 Watersheds . . . . . .
3.1.10 ifs ccl . . . . . . . . .
3.1.11 ifscfft2d . . . . . . . .
3.1.12 ifsc2imag . . . . . . .
3.1.13 ifsc2mag . . . . . . . .
3.1.14 ifsc2phase . . . . . . .
3.1.15 ifsc2real . . . . . . . .
3.1.16 ifsInsert2Dinto3D . . .
3.1.17 ifsmult . . . . . . . . .
3.1.18 ifsrecip . . . . . . . . .
3.1.19 ifssub . . . . . . . . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
ifs image
. . . . . .
. . . . . .
. . . . . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
85
86
86
86
86
86
87
87
87
88
88
88
88
89
89
90
90
90
90
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
93
93
93
93
94
94
97
98
100
101
101
101
105
106
107
108
109
110
111
112
113
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
CONTENTS
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
114
114
114
115
115
116
116
4 Image Synthesis Programs
4.1 Qsyn-synthesize range images . . . . . . . . . .
4.2 3dsyn-synthesize density images . . . . . . . . .
4.3 Matte - synthesize luminance images . . . . . .
4.4 Tomosim - simulate tomographic X-ray source .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
117
117
123
133
134
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
ifs image
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
. . . . . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
135
135
135
135
136
137
137
137
137
137
137
137
138
138
139
139
139
139
139
139
139
139
140
140
140
140
141
3.2
3.3
3.1.20 ifsvidscale . . . . . . . .
Functions to modify images . .
3.2.1 ifscolormap . . . . . . .
3.2.2 Lower Complete Images
3.2.3 zoomupdown . . . . . .
Random number generators . .
3.3.1 gaussrand . . . . . . . .
7
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
5 Programs for processing images
5.1 add . . . . . . . . . . . . . . . . . . . . . . . . . . .
5.2 addhdr . . . . . . . . . . . . . . . . . . . . . . . . .
5.3 atoi . . . . . . . . . . . . . . . . . . . . . . . . . .
5.4 any2any . . . . . . . . . . . . . . . . . . . . . . . .
5.5 c2imag . . . . . . . . . . . . . . . . . . . . . . . . .
5.6 c2mag . . . . . . . . . . . . . . . . . . . . . . . . .
5.7 c2phase . . . . . . . . . . . . . . . . . . . . . . . .
5.8 c2real . . . . . . . . . . . . . . . . . . . . . . . . .
5.9 compmag . . . . . . . . . . . . . . . . . . . . . . .
5.10 ChooseFrame . . . . . . . . . . . . . . . . . . . . .
5.11 gauss add Gaussian-distributed noise to an image .
5.12 ifsDoG Apply a derivative of Gaussian kernel to an
5.13 itoa . . . . . . . . . . . . . . . . . . . . . . . . . .
5.14 mkdoc . . . . . . . . . . . . . . . . . . . . . . . . .
5.15 mult . . . . . . . . . . . . . . . . . . . . . . . . . .
5.16 profile . . . . . . . . . . . . . . . . . . . . . . . . .
5.17 prthdr . . . . . . . . . . . . . . . . . . . . . . . . .
5.18 rmvhdr . . . . . . . . . . . . . . . . . . . . . . . .
5.19 subsample . . . . . . . . . . . . . . . . . . . . . . .
5.20 recip . . . . . . . . . . . . . . . . . . . . . . . . . .
5.21 sub . . . . . . . . . . . . . . . . . . . . . . . . . . .
5.22 triangulate . . . . . . . . . . . . . . . . . . . . . .
5.22.1 Input File Format . . . . . . . . . . . . . .
5.22.2 Output File Format . . . . . . . . . . . . .
5.23 vidscale . . . . . . . . . . . . . . . . . . . . . . . .
5.24 window . . . . . . . . . . . . . . . . . . . . . . . .
8
CONTENTS
5.25 zoomupdown . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 141
6 Programs for Analyzing Images
6.1 Harris-Laplace . . . . . . . . . . . . . . . . . . .
6.2 PCON - Piecewise-constant image noise removal
6.3 PLIN - Piecewise-linear image noise removal . . .
6.4 WaterSheds . . . . . . . . . . . . . . . . . . . . .
6.5 ccl Connected-component Labeling . . . . . . . .
6.6 resample curve . . . . . . . . . . . . . . . . . . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
143
143
143
146
148
149
151
7 Programs for Viewing IFS images
7.1 ColorMap . . . . . . . . . . . . . . . . . .
7.2 Eigenimages . . . . . . . . . . . . . . . . .
7.3 ifsview – view ifs images . . . . . . . . . .
7.4 makecolor . . . . . . . . . . . . . . . . . .
7.5 PlayMovie- play a 3D ifs image as a movie
7.6 Plotcontour . . . . . . . . . . . . . . . . .
7.7 C2LaTeX-convert source code to LaTeX .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
153
153
154
154
156
156
157
157
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
Chapter 1
The IFS system
The IFS Image File System is a set of subroutines, and programs based on those subroutines, used to manipulate images, from within C or C++ programs. An image just refers
to any array of data. The term came into use because IFS was originally written to manipulate 2-dimensional pictures, such as ones obtained from a standard camera. However,
IFS is not restricted to 2-dimensional images, and is capable of handling arrays of arbitrary
dimensionality. In the current release of IFS, most of the image manipulating routines are
designed specifically for 2- or 3-dimensional data. Later releases of IFS will have enhanced
multidimensional routines. At the simplest level, IFS is a simple system to use, and hides
from the user the implementation details of basic data manipulation functions, such as
allocating space for data arrays, performing I/O, and manipulating images with different
data formats. When used in this way, IFS provides a development tool for program writing, and is not designed to for high execution speed. Use of the standard IFS image access
functions such as ifsigp is in fact quite slow in some operations. It is efficient when the aim
is to write and test programs quickly and easily. However, for the sophisticated user, IFS
does provide access to pointers and data types which allow very speed- efficient software
to be written while retaining the IFS image structure. The FLIP (floating point image
processing) library was written in this way, to optimize computer speed.
1.0.1
New with version 8.1
There is no longer a libiptools.a All functions except flip are in libifs.a There is now a
function which will rotate images about their center, flrotate. In the flip library, however,
it will accept inputs which are data types other than float. It does floats and u8bit with
pointers, so it’s pretty fast.
9
10
CHAPTER 1. THE IFS SYSTEM
1.0.2
New with Version 8
Several changes to ifs were mandated by the changes in technology which produced a move
to 64-bit machines. These include
• Complete support for 32-bit and 64 bit machines. This has been tested on Unix
(macintosh) and Linux, but not yet on Windows.
• New interface with OpenCV. OpenCV provides many attractive features, such as the
ability to easily convert bmp, jpg, and other data types, but lacks many of the ifs
capabilities, including the ability to manipulate images with any data type and any
number of dimensions. Version 8 supports this with a new version of any2any which
pretty much will convert any data type to any data type (there are some exceptions
of course). The new OpenCV interface also provides a new image viewing capability
which smoothly handles two- and three-dimensional images and color images.
1.0.3
New with Version 6
• Support for all processing platforms. Versions of IFS release 6 run on Intel computers
under the Microsoft Windows operating system.
• IFS Win32 has been tested under windows 2000 and windows XP. It has not been
tested (and will probably not work) under Windows 3.1
• Release 6 also runs under RedHat Linux, Enterprise. It has been run under RedHat
Fedora, but has not been tested thoroughly.
• Release 6 runs on the SUN platforms under Solaris. Release 6 runs on the Macintosh
under OS-X. It does not run under Mac OS 9.
• Compression By default, all files written by IFS 6 are compressed. They are automatically uncompressed on read. It is possible for the application program to set a
flag in the header telling the write subroutine to not compress the file. Uncompressed
files can be read by IFS 5, but compressed files can only be read by IFS 6.
• New, enhanced display capabilities
1.0.4
New with Version 5
• Support for X-windows. IFS images may now be displayed on any X-windows device.
The graphics support automatically determines the dynamic range of the graphics
device many X-windows devices are binary, for example ,and either thresholds or
dithers, at the users command.
1.1. USING IFS
11
• CPU independent code. Various computers use different conventions for storage
of bytes within a word, necessitating byte-swapping when one machine reads a file
written on another. Furthermore, depending on the computer, byte swapping may
be required on 16 bit or 32 bit boundaries, or both. Finally, there are at least two
different conventions for floating point data, DEC and IEEE,which must still be
corrected after byte swapping. IFS Version 5 automatically determines what type
of machine it is running on and determines what type of machine wrote the input
file. Should the input file be incompatible, the IFS file read routine automatically
performs all data conversions.
1.1
1.1.1
Using IFS
Program Compilation and Linking
In order to use IFS, the user needs to link his/her programs to the IFS library at compilation
time. To specify these libraries in a Unix environment, one would use the switches -lifs on
the cc or ld command. For example, a typical command to compile a program would look
like:
cc -g test.c -o test -lifs
The compiler directive
-lifs
actually tells the loader to look in /usr/lib or /usr/local/lib for a file named libifs.a. Not
all computers have libifs.a installed in the /usr directory, so the user may need to change
this command. Instead of using
-lifs
simply use the entire path the the library file. For example, it might be
cc whatever.c
/myhomedirectory/ifs/ifslbiraries/libifs.a
To actually make use of the IFS functions, the users program also needs to include a header
file or two to define various structures used by the IFS routines. All programs which use
IFS should include the files stdio.h and ifs.h. Most IFS routines will return error codes
through an external global variable called ifserr; if the user plans to examine these error
codes he/she should also include the header file ifserr.h. This file defines a set of symbolic
constants which one may use rather than using actual values for codes. It is not wise to
use actual values in place of these constants when writing programs as the deffnitions for
the constants may change from one release of IFS to the next.
12
CHAPTER 1. THE IFS SYSTEM
#include<stdio.h> #include<ifs.h>
main() {
IFSIMG img1,img2; /*Declare pointers to headers */
int len[3]; /*len is an array of dimensions, used by ifscreate*/
int threshold;/*threshold is an int here */ int row,col;/*counters */
int v; img1=ifspin("infile.ifs");/*read in file by this name */
len[0]=2;/*image to be created is two dimensional */
len[1]=128;/*image has 128 columns */
len[2]=128;/*image has 128 rows */
img2=ifscreate("u8bit" ,len, IFS_CR_ALL,0) ;/*image is unsigned 8bit */
threshold=55;/*set some value to threshold */
for(row=0;row< 128;row++) for(col=0;col< 128;col++)
{
v=ifsigp(img1,row,col);/*read a pixel as an int*/
if(v<threshold)
ifsipp(img2,row, col ,255); else ifsipp(img2, row, col, 0);
} ifspot(img2,"img2.ifs"); /*write image 2 to disk */
}
Figure 1.1: An example ifs program
1.2
Referencing IFS images
All IFS images include a header which contains various items of information about the
image, such as the number of points in the image, the number of dimensions for the image,
and the data format. Also associated with the image is the actual data for the image. The
image header includes a pointer to the image data in memory. The user manipulates an
image by calling some function in the IFS library; one of the arguments to the function
will be the address of the header for the image. The functions will automatically figure
out where the data is and how to access it from the information in the header. In addition
to handling the work of accessing data in images, the IFS routines automatically take care
of allocating of space in memory to store data and headers. Everything is totally dynamic
in operation; there are no fixed-dimension arrays needed. This relieves the user of the
difficulties involved with accessing data in arrays, when using C, when the arrays are not
of some fixed size. The header structure for an image is defined in the file ifs.h, and is
known by the name IFSHDR. To manipulate an image, the user merely needs to declare a
pointer to an image header structure as
IFSHDR *yourimage;
or, equivalently,
1.3. ERROR HANDLING IN IFS
13
IFSIMG yourimage;
Then, the user simply calls some IFS function to create a new image, and sets the
pointer to the value returned from that function. Some typical programs are given in the
examples in 1.1 through ??.
1.2.1
Naming conventions in IFS
Almost all of the IFS functions have names which begin with the letters ifs, so users should
have no problems avoiding conflicts when naming their own functions. Also, all external
variables or defined constants also begin with the letters“ifs” or “IFS”. Originally, all IFS
routines had names which were limited to 6 characters in an effort to improve compatibility
between different compilers. Unfortunately, with three of the letters already being fixed
as “ifs”, this doesnt leave much left to create meaningful function names with. Hence,
many IFS functions have rather cryptic names. Later versions of IFS have relaxed this
restriction, so that newer functions have longer and more descriptive names.
Starting with release 3.0 of IFS, all of the IFS functions also have version numbers
built into them. These version numbers are actually printable strings which are globally
accessible. These strings usually contain the functions name, a version number, and the
date of the last modification to the function. Other items of information may occasionally
also be contained in the string. If an IFS function has a name ifsXXXX, where XXXX
is just some stem naming the function, then the string which gives its version number
will have the name ifsvXXXX. For instance, if a user wanted to know what version of the
function ifscreate was in his IFSlibrary, he could include the statements
extern char *ifsv_create; printf("version is \%d \n",ifsv_create);
somewhere within her program.
1.3
Error handling in IFS
IFS provides various levels of error checking. When an error occurs, an IFS function usually
returns some sort of error flag. IFS also has two external global variables which relate to
error handling. The first one is known as ifserr, and is set to error codes which the user may
examine to help determine what went wrong. The second one is IFSSLV for IFS Severity
Level, which affects the action IFS takes upon detecting an error. Both of these variables
are declared as extern int variables in the header file ifs.h, so it is not necessary for the
user to declare them.
The various error codes which may be returned are defined in the header file ifserr.h,
which the user should make sure to include in his/her program if he/she plans on using
ifserr. These error codes are described in detail in Section 1.5. Scrutinizing the file ifserr.h
may also prove useful.
14
CHAPTER 1. THE IFS SYSTEM
The error codes are indicated by individual bits in ifserr, so it is actually possible for
several error flags to be set simultaneously. Also,some error codes are actually combinations of other codes. For instance, the codes IFSENOOPEN and IFSENOTIMAGE are
two possible errors which may occur when trying to read or write IFS images. If the user
checks for the condition IFSEIOERR, she has automatically tested for both of the errors
IFSENOOPEN and IFSENOTIMAGE. The way to test for such error codes is with the bitwise logical AND operator, rather than with a comparison. I.e.: if(ifserr&IFSE IO ERR)...
is preferable to:
if(ifserr == IFSE IO ERR...
because in this way, more than one bit may be tested, or just a single bit.
The second global variable, IFSSLV, allows the user to specify what action to take
when an error occurs. Currently, there are three possible courses of action to take upon an
error;these are chosen by setting IFSSLV to some severity level code. The three severity
levels are represented by the constants IFSQUIET, IFSWARN, and IFSFATAL, which are
defined in ifs.h. These affect the action taken upon the occurrence of an error as follows:
• IFSQUIET Do not print out any error messages to the user. The function just returns
an error code to the calling routine. The user must make sure to watch out for this
code, and act accordingly. If the error is not handled, the program will probably
crash.
• IFSWARN If an error occurs, print out some message describing the error to stderr.
The routine also returns the appropriate error code. This allows the user to know
what is going on, but still allows the program to trap errors. IFSWARN is probably
the recommended severity level for most applications, and is the default value for
IFSSLV.
• IFSFATAL If an error occurs, print out an error message, and abort the program.
This is not an exceedingly user-friendly option, but is probably better than the perennial Unix favorite “buserror:core dumped”.
1.3.1
Image validation in IFS
Most IFS functions will double-check the header of an image before attempting to perform
some operation on the image. This is done to verify that the argument the user passed to
the function legitimately points to an IFS image, and does not just represent some random
value. The most likely source for such an error would be insufficient error checking in a
users program, when the severity level variable IFSSLV was set to some value other than
IFSFATAL. For instance, a section of code such as
img=ifsmkh(nrows,ncols,ubyte");
ifsipp(img,10,20,255);
1.3. ERROR HANDLING IN IFS
15
#include <stdio.h>
#include <ifs.h>
main() {
IFSIMG img1,img2; /*Declare pointers to headers */
int *len; /*len is an array of dimensions, used by ifscreate*/
int frame,row,col; /*counters */
float threshold,v; /*threshold is a float here */
img1 = ifspin("infile.ifs"); /*read in file by this name */
len = ifssiz(img1); /*get dimensions of input image */
/*ifssiz returns a pointer to an array of dimensions*/
img2 = ifscreate(img1->ifsdt, len, IFS_CR_ALL,0); /*output image is to be*/
/*same type as input */
threshold = 55; /*set some value to threshold */
/* check for one, two or three dimensions*/
switch(len[0]
{
case 1: /*1dsignal*/
{
for(col=0;col<len[1];col++)
v=ifsfgp(img1,0,col);/* read a pixel as a float*/
if(v<threshold) ifsfpp(img2,0,col,255.0);/* write a float */
/* if img2 not float, will be converted*/
else ifsfpp(img2,0,col,0.0);
}
break;
case 2:/*2dpicture*/ for(row=0;row<len[2];row++)
for(col=0;col<len[1];col++) {
v=ifsfgp(img1,row,col);/*read a pixel as a float*/
if(v<threshold)
ifsfpp(img2,row,col,255.0);/* store a float */
else ifsfpp(img2,row,col,0.0);
} break;
Figure 1.2: First portion of example, see Figure 1.3 for remainder
16
CHAPTER 1. THE IFS SYSTEM
case3:/*3dvolume*/
for(frame=0;frame<len[3];frame++)
for(row=0;row<len[2];row++)
for(col=0;col<len[1];col++)
{
v=ifsfgp3d(img1,frame,row,col);/*read a pixel as a float */ if(v<threshold
ifsfpp3d(img2, frame, row, col, 255.0); else
ifsfpp3d(img2,frame,row,col,0.0);
}
break;
default: printf(" Sorry I cant do 4 or more dimensions\n");
}
ifspot(img2,"img2.ifs"); /*write image2 to disk */
}
Figure 1.3: Example IFS program to threshold an image using number of dimensions, size
of dimen-sions, and data type determined by the input image
(Although the example above uses ifsmkh, it is now recommended that ifscreate should be
used instead.)
which attempts to create an image and set the pixel at location 10,20 to a value of 255
could be a potential source for an error, if ifsmkh had been incapable of creating the image
as requested. It would then have returned the value NULL, which would be passed to the
function ifsipp. If ifsipp did not check the header, it would blindly attempt to use NULL
as a pointer to an image header, which would probably crash the users program.
The problem with this error checking is that it takes time to perform. If an image was
100 by 100 pixels in size, and the routine ifsipp was used to set the value of each pixel
in the image, then the header would end up being checked 10000 times! For a program
which accesses an image heavily, this header checking overhead takes a significant amount
of time. Timing analyses on sample programs have shown that it is possible for 30 % of
the CPU time used by a program to be spent in the header checking operation.
The user may disable the header checking operation in some IFS routines. This, of
course, places upon the programmer the responsibility to perform more extensive error
checking operations, if robust code is desired. If the user sets the external integer variable
IFSCHK to zero, then certain routines will cease to check image headers. Header checking
can be reenabled by setting IFSCHK to any non-zero value. Note that not all of the IFS
routines are affected by IFSCHK. Generally, only those routines which are called with
high frequency, and for which the header checking represents a significant fraction of the
execution time for that function, will be affected by this variable. Incidentally, it is not
necessary for the users program to declare IFSCHK. As with IFSSLV and ifserr, this is
1.3. ERROR HANDLING IN IFS
17
#include <stdio.h>
#include <ifs.h> main()
{
IFSIMG img1,img2; /*Declare pointers to headers*/
int len[3]; /*len is an array of dimensions, used by ifscreate*/
int size; /*number of bytes in image*/
int threshold; /*threshold is an int here*/
register int count; /*number of pixels in image*/
register unsigned char *ptri,*ptro;
img1=ifspin (""); /*read in file; prompt user for name*/
len[0]=2; /*image to be created is two dimensional*/
len[1]=ifsdimen(img1,0); /*get columns of input*/
len[2]=ifsdimen(img1,1); /*get rows of input*/
img2=ifscreate("u8bit",len,IFS_CR_ALL,0); /*image is unsigned 8 bit*/
threshold=55; /*set some value to threshold*/
ptri = (unsigned char *) img1->ifsptr; /*get address of input data*/
ptro=(unsigned char *) img2->ifsptr; /*get address of output data*/
size= len[1] * len[2]; /*compute number of pixels*/
for(count=0;count<size;count++)
{
if(*ptri++ >= threshold) *ptro = 255; else
*ptro=0; ptro++; }
ifspot(img2,""); /*write image2 to disk, prompt user for filename */
}
Figure 1.4: Example: A more sophisticated version of an IFS program to threshold an
image using two dimensions, size of dimensions d and defined data type of unsigned char
on both files. Pointers are used for speed.
18
CHAPTER 1. THE IFS SYSTEM
#include <stdio.h>
#include <ifs.h>
#include <ifserr.h> /*optional*/
int main()
{
IFSHDR *img1,*img2; /*Declare pointers to headers*/
. . .
img1=ifsmkh(128,128,"char"); /*make a 128*128 2-d image*/
/* Space for data & header are automatically allocated */
....manipulateimage1
.....
img2 = ifs_exwin(img1,10,10,100,75); /*e1x2tract a sub-image*/
/*of the original image, img1, and call it img2*/
ifspot(img1,"img1.ifs"); /*writeimage1todisk */
}
declared in the header file ifs.h.
1.3.2
Header Files
The ifs user will find it convenient to include one or more of several header files, as listed
below:
ifs.h This is the main header file. Most programs which use ifs will need this file. It defines
the ifs header, and the types of many of the ifs functions.
ifstypes.h This file is useful if the user wishes to determine the data type of an image. It
defines things like IFSA U8BIT, as in the example below:
#include <ifsltypes.h>
...
if (img->dtype == IFST_32FLT)
...
which checks if the data type of image img is 32 bit float.
ifserr.h See section ??
ifsmatrix.h Defines the values returned by the matrix operation functions. See section
2.2.2
1.3. ERROR HANDLING IN IFS
1.3.3
19
Coordinate systems and array storage in IFS
IFS stores arrays in the same manner that C normally does. As with C, the indices for
arrays start with zero rather than one. For example, if you create an image with 30 rows
and 20 columns, then valid row indices for that function range from 0 to 29, and column
indices may go from 0 to 19. One common source of confusion is the usage of the terms
row and column to denote array subscripts when working with 2-dimensional arrays. It is
quite typical for a users program to viewthe coordinate system in terms of an x and a y
axis. The intent in IFS is that the column axis represents the horizontal axis, and the row
axis is the vertical. Hence the width of the image is equivalent to the number of columns
in the image, and height is the number of rows. It is common usage that the x axis is the
horizontal axis, hence a column coordinate is synonymous with an x coordinate. If this
is the coordinate system you normally use, beware of the temptation to write code of the
form:
int x,y; ... for ( x=0; x<width; x++) for( y=0;y<height; y++){ ifsipp(img, x, y, value); ... }
The correct code in this case should be:
int x,y; ... for(x=0;x<width;x++) for(y=0;y<height;y++){ ifsipp(img,y,x,value); ...}
Unfortunately, people have a tendency to write coordinate pairs as (x, y) or (row,column),
but these two are not synonymous if you interpret x, y, row,and column in the manner
described above.
A second problem occurs when displaying images on graphics output devices. There is
no set standard as to where the origin of the coordinate system is among graphics displays.
It is probably most common that the origin is in the upper left corner of the display, and
moving in the positive direction along the column axis moves you to the right, and moving
in the positive direction along the row axis moves you downwards. Some devices place
the origin in the lower left corner of the screen, and moving in the positive row direction
moves you towards the top of the display. The positive column direction still usually is
to the right. This also corresponds to the way most people label axes when hand-drawing
a graph. The net affect here is that images displayed in this coordinate system will be
upside-down as compared to the first type of system. To further confound the issue, many
programs which plot on printers reverse the meaning of the x and y axes, so that images
plotted in this manner are rotated by 90 degrees in one direction or another.
The point of these warnings about display coordinate systems is that IFS knows nothing
about the nature of the users display mechanism. There is no specific “up”, “down”, “left”,
or “right”. The user should not be too surprised if an image appears flipped or rotated
from what was expected.
20
1.4
CHAPTER 1. THE IFS SYSTEM
IFS FUNCTION LISTING
This section lists all of the functions in the IFS library, in alphabetical order. The convention used to describe the syntax for the function is:
return_value=function_name(arg1,arg2,....);
type of return_value; type of arg1; type of arg2;....
where “type of” denotes a C variable type (such as “int”, “float”, “char* ”, or “IFSHDR*”,
or other TYPEDEFs or STRUCTs). For instance, the sample description
p=ifsalc(numbytes); char *p; int numbytes;
indicates that the function ifsalc returns a pointer to a character, and that it takes one
argument, which is an integer.
1.4. IFS FUNCTION LISTING
1.4.1
21
bilininterp
//====================bilininterp================
Function to perform bilinear interpolation to find the best estimate of a pixel brightness
using the four surrounding pixels.
arguments: an ifs image, frame, row, column values
Returns the estimate.
POTENTIAL BUG: does not check if this pixel is next to the end of the image
note, this version does not interpolate in the frame direction
set frame = 0 to interpolate a 2D image
This function is part of the iptools function library, so you need to load libiptools.a in
addition to ifs.a
float bilininterp(IFSIMG inimage, int frame,float row, float col);
1.4.2
ifsalc
ifsalc – allocate storage (memory)
cptr=ifsalc(NumBytes); char *cptr; int NumBytes;
Ifalc is an IFS function used to allocate storage in main memory such as for storing arrays
and image headers. The storage will be initialized to all zeroes. It is essentially just a
call to the system function calloc; the only difference being that ifsalc performs a small
amount of error checking. If the system cannot allocate the requested amount of storage,
then ifsalc will return the value NULL, and the external variable ifserr will be set to the
value IFSENOMEM. If the external variable IFSSLV is not set to the value IFSQUIET,
then ifsalc will write an error message to stderr if it cant allocate the requested space. If
IFSSLV is set to IFSFATAL, then ifsalc will also abort your program upon an error.
22
CHAPTER 1. THE IFS SYSTEM
1.4.3
ifscigp read from complex or real image
ifscigp – get pixel value from a 2-d (possibly complex) image Usage:
val = int ifscigp(ptri,row,col) IFSHDR *ptri; /*pointer to image header structure */
int row, col, val; /*coordinates - in pixels - of pixel to examine.*/
Ifscigp returns - as an int - the value of the pixel at a specified coordinate in a 2-d image.
If image is complex format, returns the imaginary portion of the number. Known Bugs,
Special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the pixel value wont fit in an int for example, a large number in a float or complex
image ,then results are undefined. Maybe you get garbage, maybe your program
aborts on an overflow type of error.
1.4. IFS FUNCTION LISTING
1.4.4
23
ifscigp3d
ifscigp3d – get pixel value from a 3d-dimage Usage:
val = int ifscigp3d(ptri,frame,row,col);
IFSHDR *ptri; /*pointer to image header structure*/
int frame, row, col; /*coordinates in pixels of pixel to examine.*/
Ifscigp3d returns as an integer the value of the pixel at a specified coordinate in a 3-d
image. If image is complex format, returns the imaginary portion of the number assuming
it can be converted to an int. Known Bugs, Special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the pixel value wont fit in an int for example, a large number in a float or complex
image then results are undefined. Maybe you get garbage, maybe your program
aborts on an overflow type of error.
24
CHAPTER 1. THE IFS SYSTEM
1.4.5
ifscipp
IFS library:
libifs.a
ifscipp: set pixel value in a 2-d image usage:
status = ifscipp(ptri,x,y,val);
IFSHDR *ptri; /*pointer to image headers tructure*/
int x,y; /*coordinates -- in pixels -- of pixel to examine.*/
int val;
int status; /*return status*/
Returns: IFSSUCCESS or IFSFAILURE
Ifscipp sets the value of the pixel at a specified coordinate in a 2-d image, where the
input is an int. If image is complex format, stuffs the imaginary portion of the number,
and DOES NOT set the real part to zero.
Known Bugs, Special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the pixel value wont fit in an int for example, a large number in a float or complex
image then results are undefined. Maybe you get garbage, maybe your program
aborts on an overflow type of error.
1.4. IFS FUNCTION LISTING
1.4.6
25
ifscipp3d
ifscipp3d: set pixel in a 3-d image usage:
status = ifscipp3d(ptri, frame, row, col, val);
IFSHDR *ptri; int frame, row, col; /*coordinates in pixels of pixel to examine.*/
int status; /*return status*/
Returns: IFSSUCCESS or IFSFAILURE
Ifscipp sets the value of the pixel at a specified coordinate in a 2-d image, where the
input is a int. If image is complex format, stuffs the imaginary portion of the number, and
DOES NOT set the real part to zero.
Known bugs, special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the value stored wont fit in the output image datatype, then results are undefined.
Maybe you get garbage, maybe your program aborts on an overflow type of error.
26
CHAPTER 1. THE IFS SYSTEM
1.4.7
ifscfgp
ifscfgp: get value of a pixel in a 2-d image. usage:
val - double ifscfgp(ptri,row,col); IFSHDR *ptri;
int row,col; /*coordinates (in pixels) of pixel to examine.*/
Ifscfgp returns (as a float) the value of the pixel at a specified coordinate in a 2-d image.
If image is complex format, returns the imaginary portion of the number. Known bugs,
special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the pixel value wont fit in a double results are undefined. Maybe you get garbage,
maybe your program aborts on an overflow type of error. There could be possible
round off errors.
1.4. IFS FUNCTION LISTING
1.4.8
27
ifscfgp3d
ifscfgp3d: get value of a pixel in a 3-d image usage:
val = (double) ifscfgp3d(ptri,frame,row,col); IFSHDR *ptri;
int frame, row, col; /*coordinates (in pixels) of pixel to examine.*/
ifscfgp3d returns (as a double) the value of the pixel at a specified coordinate in a 3-d
image. If image is complex format, returns the imaginary portion of the number.
Known bugs, special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the pixel value wont fit in a double results are undefined. Maybe you get garbage,
maybe your program aborts on an overflow type of error. There could be possible
round off errors.
28
CHAPTER 1. THE IFS SYSTEM
1.4.9
ifscfpp
ifscfpp: set value of a pixel in a 2-d image. usage:
status = ifscfpp(ptri, x, y, val);
IFSHDR *ptri;
int x,y; /*coordinates (in pixels) of pixel to examine.*/
double val; /*the value to stuff.*/
int status;
Returns: IFSSUCCESS or IFSFAILURE
Ifscfpp sets the value of the pixel at a specified coordinate in a 2-d image, where the
input is a float. If image is complex format, stuffs the imaginary portion of the number.
Known Bugs, Special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the value stored wont fit in the output image datatype, then results are undefined.
Maybe you get garbage, maybe your program aborts on an overflow
1.4. IFS FUNCTION LISTING
1.4.10
29
ifscfpp3d
ifscfpp3d: sets the value of a pixel in a 3-d image (This is a completely new version of
ifsfpp which handles 3-d images.) usage:
status = ifscfpp3d(ptri, frame, row, col, val);
IFSHDR *ptri;
int frame, row, col; /*coordinates (in pixels) of pixel to examine.*/
double val; /*the value to stuff.*/
int status;
Returns: IFSSUCCESS or IFSFAILURE I fscfpp3d sets the value of the pixel at a
specified coordinate in a 3-d image, where the input is a float. If image is complex format,
stuffs the imaginary portion of the number.
Known Bugs, Special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the value stuffed wont fit in the output image datatype, then results are undefined.
Maybe you get garbage, maybe your program aborts on an overflow type of error.
30
1.4.11
CHAPTER 1. THE IFS SYSTEM
Ifsclose
ifsclose: close an open file Usage:
rc = ifsClose(File);
FILE *File;
int rc;
Ifsclose, an ifs internal function seldom used by typical programmers, is identical to the
standard I/O library function fclose, except that it will avoid closing File if File corresponds
to stdin, stdout, or stderr. If File is NULL, ifsclose returns -1, else it just returns whatever
value fclose would return. Ifsclose is supplied as a complement to ifsOpen since the latter
function may return stdin or stdout in some circumstances, and the user typically does not
want to close these files.
1.4. IFS FUNCTION LISTING
1.4.12
31
ifscreate
ifscreate: create an IFS image Usage:
img = ifscreate(type, len, flags, structsize);
IFSHDR *img;
char *type;
int len[];
int flags;
int structsize;
Ifscreate is used to create a new IFS image or image header. Space for the header is
automatically allocated, and a pointer to the header is returned. Various fields in the header
structure will be set to default values. Space for the actual data may also be allocated,
depending on the value of the flags variable. If space for the data array is allocated, it will
be filled with zeros. If the image can not be created, ifscreate returns the value NULL,
and the external variable ifserr will be set to some error code, as given in the #include
file ifserr.h . The image as created will not have any tail structure associated with it. The
arguments to ifscreate are:
type The data format for individual pixels, such as byte or double. The valid data types
are listed in a later section of this manual. If the data type is not recognized by
IFS, then ifscreate will return NULL, and ifserr will be set to the code IFSE BAD
DTYPE.
len An n+1-length integer array the first element (len[0]) gives the number of dimensions
for the image, the remaining elements give the length for each dimension of the
image being created. This is in exactly the same format as the arrays returned by
the function ifssiz. The lengths are given in terms of ascending rank for the image.
Images are stored in standard C storage order: the column or x index changes most
rapidly when scanning through memory, hence this dimension has rank 1. The row
or y index has rank 2, the frame or z index has rank 3, and so on. I.e., the second
element of the array (len[1]) gives the number of columns of the image, len[2] is the
number of rows, etc.
flags The various bits of this argument determine precisely what is and is not allocated
when generating the image. If flags = IFS CR ALL or IFS CR DATA, then storage
space for the image is allocated, as well as storage for the image header. In this case
the field img->ifsptr points to the data storage. If flags = IFS CR HDR then only
space for the image header is allocated. The field img->ifsptr will be set to NULL.
The user must supply an array to store the image in, and set img->ifsptr to point to
this array. Note: versions 4.0, 5.0, and 6 of ifscreate will ALWAYS allocate space for
32
CHAPTER 1. THE IFS SYSTEM
the image header; the flag IFS CR HDR is not really examined, and is only intended
for possible future expansion. All that is really checked is the IFS CR DATA bit.
The flag IFS CR ALL is the combination of IFS CR DATA and IFS CR HDR and is
probably the best flag to use if one wants data space allocated.
structsize This argument is only needed if type is struct, in which case it gives the size of
a single data element (structure) in bytes. If type is not struct this argument should
be set to 0.
Example of using ifscreate with a data type of struct:
/* Create a 2-d image with 20 rows and 30 columns */ /* and a 1-d array of 10 structures
#include <ifs.h>
main()
{ IFSHDR * img, * strimg;
int lengths[3]; IFSHDR * ifscreate();
typedef struct { int red; int green; int blue; } RGB; . . . /* create 2D byte array */
lengths[0] = 2; /* Image will be 2D */
lengths[1] = 30; /* Number of columns (width; x-dimension) */
lengths[2] = 20; /* Number of rows (height; y-dimension) */
img = ifscreate("ubyte", lengths, IFS_CR_ALL);
if (img == NULL) { /* error processing code */
} . . . /* create 1D structure array */ lengths[0] = 1;
lengths[1] = 10;
strimg = ifscreate("struct",lengths,IFS_CR_ALL,sizeof(RGB)); . .
.
1.4. IFS FUNCTION LISTING
33
Routines which are compatible with openCV
In order to make use of the capabilities of openCV, and still retain the usefulness of IFS,
several new functions are available to go between the two formats. openCV uses an image
format called IPL, which is very similar to ifs in many ways. Most openCV functions do
not support all the data formats that ipl supports, and ipl does not support some of the
data types that ifs supports. In particular, the data types complex short, and complex are
supported by ifs, but not by ipl.
1.4.13
ifs2ipl
Converts an ifs image to an ipl-format image of the same data type. It returns a pointer
to an ipl image
IplImage *ifs2ipl(IFSIMG inputimage);
Note that the old image is NOT deleted, but a new image is created using memory
allocation. This is a potential memory leak unless the user is careful. If the old ifs image
is no longer needed, it can be deleted using
ifsfree(oldimage,IFS_FR_ALL);
which will delete the image, and the header and the tail (if there is a tail).
1.4.14
ipl2ifs
Converts an ipl image to an ifs-format image. It returns a pointer to an ifs image
IFSIMG ipl2ifs(IplImage
*inputimage);
Note that the old image is NOT deleted, but a new image is created using memory
allocation. This is a potential memory leak unless the user is careful. If the old image is
no longer needed, it can be deleted using
cvReleaseImage(&imagepointer);
1.4.15
ifsCVpin
Reads an image from the disk. If it is an ifs image already, The ifs format will be retained
in memory. If it is
• Windows bitmaps - BMP, DIB,
34
CHAPTER 1. THE IFS SYSTEM
• JPEG files - JPEG, JPG, JPE,
• Portable Network Graphics - PNG,
• Portable image format - PBM, PGM, PPM,
• Sun rasters - SR, RAS, or
• TIFF files - TIFF, TIF
it will be converted to ifs and stored in ifs format in memory.
IFSIMG ifsCVpin(char *filename);
1.4.16
ifsCVpot
Writes an ifs image to disk. If the target file name has the extension “ifs”, it will be written
retaining the original ifs format. If the target is any of the other image types, the image
MUST be unsigned char, since that is the only format supported by the ipl write software.
int ifsCVpot (IFSIMG imagepointer, char *filename);
1.4.17
ifsCVPopup
This will pop up a window on the user’s screen with an image in it. ifsCVPopup is also
usable as a general-purpose image viewing program. If you don’t want to use it as a subroutine (and have to deal with linking to openCV), you can get the same functionality
using the program ifsview.
IplImage *ifsCVPopup(IFSIMG imagepointer, int colorflag,int ignoreflag, int scaleflag );
The colorflag, if 1, means that a 3D, 3frame image should be interpreted as color rather
than as a 3 frame image. example use, put up an image, wait for a user interaction, and
close the screen window. The window is named “img1”. The ignore flag argument only
affects contrast enhancement. If it is greater than zero, than any pixel of that brightness
will not enter the calculation of contrast stretching. This is useful if you want to look at
an image containing artificially added graphics with a special brightness, like 255.
The scaleflag only affects 3D images. if scaleflag is one, each frame will be video scaled
independenlty, if zero, the entire image will brightness scaled the same.
IFSIMG iggle;
IplImage *oggle;
oggle = ifsCVPopup(iggle,0,-1);// put up the image, don’t ignore anything
1.4. IFS FUNCTION LISTING
35
cvReleaseImage(&oggle);// free the memory used by the temporary creation of
an image to display
cvWaitKey(0);//wait for the user to hit a key in the display window
cvDestroyWindow("img1"); //close the window on the screen
Note the ampersand in the call to cvReleaseImage. Yes, it is a pointer to a pointer. The
pointer itself will be freed as well as the image. The image will stay on the screen until the
user closes it, or the program exits.
CAUTION: When the file is converted from ifs to ipl, the temporary ipl image is not
deleted, since the use may want it. This is a potential memory leak if the user does not
delete the temporary image after she is finished displaying it. As illustrated above, IPL
images can be freed using
cvReleaseImage( &imagepointer);
Using a poped-up Image:
Once you have called ifsCVPopup, you have a variety of options:
• LOUPE As soon as the image pops up, and you move the mouse over it, another
image will pop up which is a zoomed1 version of the original image, zoomed about
where the mouse is. We will refer to this second image as the “loupe” image, as in a
jewler’s loupe. As you move the mouse, the second image will move around also.
• Stopping the tracking A right click on the original image will freeze the loupe image. A
second right click will turn tracking back on. On the mac, a right click is accomplished
by hitting the trackpad with two fingers simultaneously.
• Additional zoom With the pointer on the original image, you can use right and left
arrow keys to increase or decrease zooming (respectively). Note that you will not see
the result of the zoom until you move the mouse again, which will refresh the loupe
image. Note: some versions of Linux are not compatible with each other as far as
the code used to represent an arrow key. For that reason, the following alternative
keys may be used:
– i uparrow
– m downarrow
– j leftarrow
– k rightarrow
– return key return to the calling program
1
Well, it is in the scale of the original image. If the original image is 4000 × 4000, the loupe image will
seem significantly zoomed. However, it is is 512 × 512, no zoom will be apparent
36
CHAPTER 1. THE IFS SYSTEM
• Image values – original image Clicking on the original image will return the brightness
at the point of the click, or if color, will return the R,G,B values.
• Image values – loupe image To look at points in great detail, first move the mouse
to get to where you want to be in the original image. Then right click to turn off
tracking. Move the pointer to the loupe image, and click on any point. The brightness
or color of that point will be returned.
• Exploring 3D images With the mouse on the original image (tracking can be on or
off), the up and down arrow will display different frames of the 3D image. Remember
that these values don’t show up in the loupe image until you move the mouse.
• Enhancing contrast The lower-case c key will toggle contrast enhancement on or off.
When on, the loupe image will be displayed with enhanced contrast.
• Returning If you hit control C, you will be returned to the top (system) level. If you
hit return, you will be returned to whatever program call the ifsCVPopup function.
If that program was ifsview, it will, in turn, return you to the system.
Should you simply wish to display an image and obtain information about it, use the
program ifsview, which is just a wrapper around ifsCVPopup, and that way, you won’t
need to mess with linking openCV, which can be a painful hassle.
1.4.18
IFSDISPLAY
The module IFSDISPLAY in the directory iptools contains two important subroutines that
allow you to display images. int CreateIFSDisplayWindow and WriteToIFSDisplayWindow. The first of these creates a window into which you can put an ifs image and display
it. The second allow you to rewrite what is in that display window.
Usage:
int CreateIFSDisplayWindow(IFSIMG img,float zoom);
Zoom is a scale factor, which can be used to make the displayed image larger. The integer
returned is an index allowing the user to display more than one image at the same time.
A maximum of ten images may be simultaneously displayed.
int WriteToIFSDisplayWindow(int whichwindow,IFSIMG img,int frame,float zoom);
This function allows the user to overwrite the contents of a particular screen. The dimensions of the new image must match those of the original.
1.4. IFS FUNCTION LISTING
1.4.19
37
Building ifsCV and IFSDisplay with openCV
You can compile and load the openCV routines listed above using the same operations that
you used to build ifs, except you need to tell the linker to load the CV subroutines. Here
is a compile example that will do this in xcode. (add this item to the build instructions)
OTHER_LDFLAGS = -lcxcore /Users/wes/src/ifs/MacX/ifslib/libifs.a -lcvaux -lhighgui -lcv
If you are using a makefile, the compile command would be something like
cc -o myprogram -lxcore Users/wes/src/ifs/MacX/ifslib/libifs.a -lcvaux -lhighgui -lcv
This assumes, of course, that the openCV libraries were installed in the standard places
(/usr/local/lib), and have the standard names (libhighgui.a rather than libhighgui.2.3.a)
If you didn’t install them there, you will need to use the full path name, instead of the
shortened path, and if the version number is included with the library file names, you will
need to use it. It’s simpler to just use ifsview.
38
1.4.20
CHAPTER 1. THE IFS SYSTEM
ifsdimen
ifsdimen: get size of dimensionUsage:
len = ifsdimen(image, n);
int len;
IFSHDR*image;
int n;
Ifsdimen returns the length (number of elements) of the nth dimension of image. It also
may be used to get the total number of elements or bytes required by the data section of an
image. The argument n is the rank of the dimension being queried, i.e., ifsdimen(img,0) is
the number of columns, ifsdimen(img,1) is the number of rows, and so on. If n is specified
as -1, ifsdimen returns the total number of elements in the image (the product of all the
individual dimension lengths). If n is specified as -2, ifsdimen returns the total number of
bytes occupied by the image data, i.e., the total number of elements times the size in bytes
for a single element. If there is some error, ifsdimen returns zero and sets the external
variable ifserr appropriately. Possible error conditions are IFSE BAD HEADER or IFSE
NULL HEADER for invalid images, or IFSE WRONG NDIM if n is invalid (such as asking
for the number of frames for a 2D image).
1.4. IFS FUNCTION LISTING
1.4.21
39
ifsexwin
ifsexwin: Extract a window from an image Usage:
new = ifsexwin(old, r1, c1, r2, c2); IFSHDR *new, *old; int r1, c1, r2, c2;
Ifsexwin is used to create a new image which is a subimage of some old image The old
image must be a two-dimensional image. The arguments r1,c1 and r2,c2 give the row and
column positions of the corners of a box which defines the region to be extracted. These
corners must be on opposite ends of a diagonal for the window It does not matter which
corners are chosen for each point, as long as as they are on opposite ends of a box diagonal.
The region extracted includes the area of the bounding box itself, ie, is inclusive of the
rows r1, r2 and columns c1, c2. Ifsexwin returns a pointer to the newly created image, or
NULL if some error occurred. In the latter case, the external variable ifserr will be set to
indicate the nature of the error. Possibilities are:
• IFSE BAD HDR If the pointer old does not point to a valid IFS image.
• IFSE NO MEM If space couldnt be allocated for the new image.
• IFSE WRONG NDIM If the original image is not two-dimensional.
• IFSE BAD POS If either of the box coordinates is outside the image dimensions.
The dimensionality of windowed image is consistent. That is, a 1-d/2-d slice ( a 3-d image
one voxel thick in one or more dimensions) returns with a header consistent with the actual
dimensionality.)
40
1.4.22
CHAPTER 1. THE IFS SYSTEM
ifsexwin3d
ifsexwin3d: Extract a window from an image usage:
newimg = ifsexwin3d(oldimg,f1,r1,c1,f2,r2,c2)
where f1,r1,c1 and f2,r2,c2 are the coordinates (frame,row,col) of one corner of the box
and the opposite (diagonal) corner. It doesnt matter which corners are chosen.The box
which is extracted includes the bordering surface (i.e, coordinates are f1,r1,c1 to f2,r2,c2
INCLUSIVE). Returns: This function returns NULL if an error occurs, and returns an
error code thru the external variable ifserr. External variables: ifserr, IFSSLV Ifsexwin3d
extracts a piece (window) out of a 3-d IFS image, to make a new IFS image. The data
type of the new image is identical to that of the old one. The dimensionality of windowed
image is consistent. That is, a 1-d/2-d slice ( a 3-d image one voxel thick in one or more
dimensions) returns with a header consistent with the actual dimensionality.
1.4. IFS FUNCTION LISTING
1.4.23
41
ifsfgp
ifsfgp: get pixel from a 2-D image Usage:
value = ifsfgp(img, row, col);
double value;
int row,col;
IFSHDR *img;
Ifsfgp is used to get the value of some pixel in a 2-dimensional image. The value returned
is of type double, regardless of what the data format of the image is. Otherwise, ifsfgp is
identical to the function ifsigp, in all respects. See the documentation for ifsigp for more
details.
42
CHAPTER 1. THE IFS SYSTEM
1.4.24
ifsfgp3d
ifsfgp3d: gets the value of a pixel in a 3-d image (A generic multidimensional fgp can be
attempted thru variable parameter passing, but that would make the code unportable.)
usage:
val = ifsfgp3d(ptri, frame, row, col);
IFSHDR *ptri; /*pointer to image header structure*/
int frame, row, col; /*coordinates (in pixels) of pixel to examine.*/
double val;
Ifsfgp3d returns (as a double) the value of the pixel at a specified coordinate in a 3-d
image. If image is complex format, returns the real portion of the number.
Known Bugs, Special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the pixel value wont fit in a double then results are undefined. Maybe you get
garbage, maybe your program aborts on an overflow type of error. Round off error
can occur in conversions. e.g. int to double typecasting.
1.4. IFS FUNCTION LISTING
1.4.25
43
ifsfpp3d
ifsfpp3d: set the value of a pixel in a 3-d image usage:
status = ifsfpp3d(ptri, frame, row, col, val);
\index{ifsfpp3d}\index{put pixel}
IFSHDR *ptri; /*pointer to image header structure*/
int frame, row, col; /*coordinates (in pixels) of pixel to examine.*/
double val; /*the value to stuff*/
int status;
Returns: IFSSUCCESS or IFSFAILURE
Ifsfpp3d sets the value of the pixel at a specified coordinate in a 3-d image, where the
input is a float. If image is “complex” format, stuffs the real portion of the number.
Known Bugs, Special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the value stuffed wont fit in the output image data type, then results are undefined.
Maybe you get garbage, maybe your program aborts on an overflow type of error.
44
CHAPTER 1. THE IFS SYSTEM
1.4.26
ifsfpp
ifsfpp: put pixel value into a 2-D image status = ifsfpp(img, row, col, value); int status;
double value; int row, col; IFSHDR *img; Ifsfpp3d sets the value of the pixel at a specified
coordinate in a 3-d image, where the input is a float. If image is complex format, stuffs
the real portion of the number.
Known Bugs, Special notes:
• This routine does not check to see if the specified coordinates actually are in bounds.
• If the value stuffed wont fit in the output image datatype, then results are undefined.
Maybe you get garbage, maybe your program aborts on an overflow type of error.
1.4. IFS FUNCTION LISTING
1.4.27
45
ifsfree
ifsfree: delete and deallocate an IFS image img = ifsfree(img,flags); IFSHDR *img; int
flags; Ifsfree is used to get rid of an IFS image which is no longer in use. The space for
the header and/or data is deallocated, and returned to the operating system for other use.
Basically, ifsfree just consists of several calls to the system function cfree. The arguments
to ifsfree are:
• img A pointer to the image header structure.
• flags A set of flags which indicates exactly what is to be deallocated. Possibilities for
the flags are:
– IFS FR DATA If this flag is set, then ifsfree will deallocate the space allocated
for the storage of the actual image data (if there is any), and the header field
img->ifsptr will be set to NULL to show that the header no longer has any data
associated with it. If there is no data associated with the header, then this flag
has no effect. This will not cause any errors.
– IFS FR HDR If this flag is set, then ifsfree will deallocate the space allotted for
the image header. The data space is left intact. This is usually only going to
be used if the user supplied his own data area for the image (perhaps a static
array or somesuch).
– IFS FR ALL If this flag is specified, then ifsfree will free up everything image
header and data. IFS FR ALL is just the combination of IFS FR DATA and IFS
FR HDR. This is probably the normal flag to set when calling ifsfree . Ifsfree
returns a pointer to the image header as it should be AFTER the desired things
have been deallocated. If only the IFS
– FR DATA flag was specified, then ifsfree returns the original pointer value img,
with the field img->ifsptr now set to NULL to show that the data array has
been deleted. If the header structure was freed, then ifsfree returns NULL to
indicate that the pointer is no longer valid. Hence it is good practice to assign
the return value from ifsfree back to the original pointer value img. It is not an
error to simply say ifsfree(img,IFS FR ALL) rather than img = ifsfree(img,IFS
FR ALL) to get rid of an image, but the latter usage is preferable in that it will
make it more obvious to any subsequent routines called (erroneously) with the
argument img. Ifsfree will also return the value NULL if an error occurred. In
this case, the external variable ifserr will be set to the appropriate error code.
Possible error conditions are:
• IFSE NULL HDR This indicates that you passed the pointer NULL for the argument
img.
46
CHAPTER 1. THE IFS SYSTEM
• IFSE BAD HDR This indicates that the pointer img does not reference a valid IFS
image structure. Note that the error IFSE NULL HDR is actually a subclass of the
error IFSE BAD HDR, so if you test the value (ifserr & IFSE BAD HDR), you will
automatically also pick up errors of the type
• IFSE NULL HDR.
BUGS (features):
Trying to deallocate something which was not originally obtained using some standard
system memory-allocation function (e.g, calloc, or the IFS routines ifsalc or ifscreate)
will cause grave errors usually a program crash. This is a problem of the system
allocate/deallocate routines and not ifsfree.
1.4. IFS FUNCTION LISTING
1.4.28
47
ifsGetFN
ifsGetFN: read in a filename and expand it
Usage:
FileName = ifsGetFN(Prompt,Input);
char *FileName;
char *Prompt;
FILE *Input;
IfsGetFN will read in a string from the file Input (typically stdin), and expand it using the
function ifsPrsFN. It returns a pointer to the name it read, or NULL if it failed. Space
for the filename is dynamically allocated and may be freed (using cfree) when the user
is through with it. If Input is a terminal, and Prompt is not NULL, then Prompt will
be printed before the filename is read in. IfsGetFN normally delimits filenames with any
control character or whitespace character and strips off any leading whitespace characters
supplied in the name. Any character (including whitespace characters) may be put in a
filename by prefixing it with a backslash n. This applies to leading whitespace characters
as well as whitespace characters in the middle or end of the name.
48
1.4.29
CHAPTER 1. THE IFS SYSTEM
ifsGetImg
ifsGetImg: open a file and read an IFS image from it
Usage:
img = ifsGetImg(FileName, Prompt, ReadTail);
IFSHDR *img;
char *FileName;
char *Prompt;
int ReadTail;
IfsGetImg reads an IFS image from some file FileName. If ReadTail is false (zero), then any
tail information associated with the image will not be read in. It returns a pointer to the
new image header, or NULL if it failed, in which case the external integer variables ifserr
and column can be examined to determine the nature of the error. Space for the header,
data, and tail is allocated dynamically, each may be freed (using cfree or ifsfree) when the
user is through with it. If FileName is NULL, then the input image is read from stdin.
If FileName is a null string (that is, FileName = ), then a filename will be read in from
stdin using the routine ifsGetFN. In this case, if Prompt isnt NULL and stdin corresponds
to a terminal, then the string Prompt will be printed on the terminal (actually, to stderr)
before reading a name. If stdin is not attached to a terminal, such as when input is being
piped in from another program, then the printing of the prompt string is suppressed. If a
filename is read in from stdin, and it is (a single dash), then the image itself will be read
from stdin. Filenames are expanded using ifsPrsFN, so they may contain such things as
environment variable names and login id constructs.
IfsGetImg works by opening the specified file and then calling ifsRdImg to do the actual
work of getting the image. It then closes the file when its done (unless it read from stdin).
Note that calling ifsGetImg with Filename = NULL is essentially the same as calling
ifsRdImg directly. The complimentary routine to ifsGetImg is ifsPutImg.
1.4. IFS FUNCTION LISTING
1.4.30
49
ifsigp
ifsigp: get pixel from a 2-D image
Usage:
value = ifsigp(img,row,col);
int value;
int row,col;
IFSHDR *img;
Ifsigp is used to get the value of some pixel in a 2-dimensional image. The value returned is
of type int, regardless of what the data format of the image is. Ifsigp performs all necessary
type conversions. If the value of the pixel in the image will not fit into an int data type,
then the value that is returned will be meaningless. If the image data format is one of the
complex forms, then ifsigp returns the real part of the specified data point. If some sort
of error occurs, then ifsigp will return zero, and the external variable ifserr will be set to
indicate the nature of the error.
The arguments to ifsigp are:
• img A pointer to the image header structure. This should refer to a 2-dimensional
image. If the image has 3 or more dimensions, then ifsigp will access the first frame
of the data (ie, all indices besides the first two will simply be treated as zero).
• row,col The coordinates of the point to be examined. Note that row, col may also be
regarded as a y, x pair. Beware that row corresponds to the y index, not the x index
. The following error codes (defined in the #include file <ifserr.h >) may be returned by
ifsigp :
• IFSE BAD HEADER: The pointer img does not point to an actual IFS image.
• IFSE BAD DTYPE: The image is of some data type that ifsigp does not recognize.
Usually this indicates that your header has been damaged, and the field img ! ifsdt
is mangled, or the image data type is struct. It could also occur if someone added a
new data type to that understood by IFS, and forgot to modify ifsigp accordingly.
BUGS (features): Ifsigp does not verify that the image passed to it corresponds to a
2-dimensional image.
The indices row, col may (depends on severity level setting) not checked to verify that
they lie inside the image dimensions. Ifsigp does not check to make sure that the data
pointer img-/textgreaterifsptr is not NULL. Previous versions of IFS did not allow this
data pointer to be NULL, so it was not previously necessary to check for this.
Results when numeric overflow occur (as is possible when converting a floating point
number into an integer) are undefined.
50
CHAPTER 1. THE IFS SYSTEM
Any of the above issues could cause an abrupt and unpleasant termination of your
program, generally with the infamous “bus error: core dumped” message under UNIX
systems. Of course, such a crash would be indicative of some prior error in the users
program not having been caught.
1.4. IFS FUNCTION LISTING
1.4.31
51
ifsipp
ifsipp: put pixel value into a 2-D image
status = ifsipp(img,row,col,value);
int status;
int value;
int row,col;
IFSHDR *img;
Ifsipp is used to set the value of some pixel in a 2-dimensional image. The argument value
is automatically converted from an integer into whatever data format the image is in. If
the image is of type complex, then ifsipp sets the real part of the datum to value, and the
imaginary portion to zero.
Ifsipp returns the value IFS SUCCESS if it succeeded, otherwise it returns the value
IFS FAILURE and sets the external variable ifserr to the appropriate error code.
The arguments to ifsipp are:
• img A pointer to the image header structure. This should refer to a 2-dimensional
image. If the image has 3 or more dimensions, then ifsipp will access the first frame
of the data (ie, all indices besides the first two will simply be treated as zero). row,col
The coordinates of the point to be examined. Note that row, col may also be regarded
as a y, x pair. Beware that row corresponds to the y index, not the x index.
• value The actual data value to be put into the image. Note that if the datum
represents some value that can not be represented in the data format of the image
itself (such as trying to place the value 500 into a ubyte image), a meaningless value
will end up being put into the image.
The following error codes (defined in the #include file ifserr.h ) may be set by ifsipp (in
the variable ifserr):
• IFSE BAD HEADER: The pointer img does not point to an actual IFS image.
• IFSE BAD DTYPE: The image is of some data type that ifsipp does not recognize.
Usually this indicates that your header has been damaged, and the field img->ifsdt is
mangled, or that the image data type is “struct”. It could also occur if someone added
a new data type to that understood by IFS, and forgot to modify ifsipp accordingly.
BUGS (features):
• Ifsipp does not verify that the image passed to it corresponds to a 2-dimensional
image.
• The indices row, col are not checked to verify that they lie inside the image dimensions.
52
CHAPTER 1. THE IFS SYSTEM
• Ifsipp does not check to make sure that the data pointer img-/textgreaterifsptr is not
NULL. Previous versions of IFS did not allow this data pointer to be NULL, so it
was not previously necessary to check for this. Any of the above issues could cause
an abrupt and unpleasant termination of your program, generally with the infamous
“bus error: core dumped” message under UNIX systems. These problems will not
occur however, unless the users program contains some other sort of error.
1.4. IFS FUNCTION LISTING
1.4.32
53
ifsmkh
ifsmkh: Create a two-dimensional IFS image THIS FUNCTION IS OBSOLETE STARTING WITH RELEASE 3.0 OF IFS THE FUNCTION ifscreate SHOULD BE USED INSTEAD
54
1.4.33
CHAPTER 1. THE IFS SYSTEM
ifsopen
ifsopen: open a file for reading or writing.
Usage:
File = ifsOpen(FileName,Mode,Prompt,NumRetries);
FILE *File; char *FileName;
char *Mode;
char *Prompt;
int NumRetries;
Ifsopen opens up a file FileName for reading or writing, and returns a pointer to the open
file descriptor (stream in Unix terminology).
Ifsopen is used by ifspin, and the usual user does not (normally) need it.
If the file can not be opened or some other error occurs, then ifsopen will return NULL.
The argument Mode is the same as the mode argument to the standard i/o library function
fopen, i.e. r or w for read or write access, (on Windows - based compilers, one sometimes
must use “rb” and “wb”. If FileName is NULL, then ifsopen just returns stdin or stdout
as is appropriate for the specified Mode.
If FileName is a null string (FileName = “”), then ifsopen will read the name of the
file to be opened from stdin. If stdin is attached to a terminal, then the string Prompt
will be printed before getting the filename (unless Prompt is NULL). FileName is read
using the function ifsGetFN, and expanded using ifsPrsFN, so it may contain the names
of environment variables or constructs of the form login id to represent some users home
directory name. If the name read in is a single dash, then ifsopen will return stdin or
stdout, according to the argument Mode.
If a filename is being read interactively (when FileName = “”, stdin is connected to a
terminal, and Prompt is not NULL), then the user is allowed NumRetries mistakes before
ifsopen will give up and return NULL. For instance, if ifsopen tries to open a non-existent
file for reading, it will print a message to the user and ask for a new name. After several
failures it will give up. This is to prevent such things as runaway shell scripts from sitting
in a perpetual error loop.
1.4. IFS FUNCTION LISTING
1.4.34
55
ifspin
ifspin: read in an image from disk
img = ifspin(filename);
IFSHDR *img;
char *filename;
Ifspin is used to read an IFS image from the specified file filename. All necessary storage
space for the image and its data is automatically allocated. The “tail” information for the
file is not read in. If the user wants the tail information read in, he should use the newer
function ifsGetImg. If filename points to a null string, then ifspin will prompt the user to
specify some filename. Any filename (whether or not read interactively) will be translated
using the function ifsPrsFN, which will substitute for environment variables and names of
users home directories specified in the C- shell shorthand form of user/filename. If filename
is NULL, then input will be read from stdin. Also, if a user is prompted for a filename,
if she specifies a single dash, the input will be read from stdin. The printing of a prompt
string will be suppressed if stdin is not attached to a terminal. Ifspin returns a pointer
to the new image, or NULL if some sort of error occurs. In the latter case, the external
variable ifserr will be set to indicate the nature of the error. Possibilities are:
• IFSE NO OPEN if the specified file cant be opened (usually meaning that it doesnt
exist.
• IFSE IO ERR if some sort of I/O error occurred (usually meaning the file does not
contain a valid IFS image). The standard system I/O library variable errno may
contain additional information about the nature of the error. Note that IFSE NO
OPEN is a subclass of the IFSE IO ERR error, so one can check for both automatically
by using a construct of the form “if (ifserr & IFSE IO ERR) action to take();”
• FSE NO MEM if it isnt possible to allocate storage to put the image into. IFSE
BAD NAME if some error occurred when translating the file name.
BUGS/NOTES:
Ifspin is an obsolete function. Under version 4 of IFS, this just remaps its arguments
and calls ifsGetImg. Even though it is obsolete, it is easier to use than ifsGetImg, many
programs use it all the time.
56
CHAPTER 1. THE IFS SYSTEM
1.4.35
ifspot
ifspot: write an image to disk Usage:
status = ifspot(img,filename);
int status;
IFSHDR *img;
char *filename;
Ifspot is used to write an IFS image to the specified file filename. If filename points to a
null string, then ifspot will prompt the user to specify some filename, and read a filename
from stdin. If filename is NULL, then ifspot will write the image to stdout. Also, if ifspot
reads a filename from stdin, and the filename is -, then ifspot will write to stdout. If stdin
is not connected to a terminal (e.g, input is being piped in from another program), then
the printing of a prompt will be suppressed.
The filename is translated using ifsPrsFN, so it may contain environment variables
(beginning with a leading “$”) and the names of users home directories specified in the
C-shell shorthand form of “user/ filename”.
Ifspot returns the value IFS SUCCESS if it succeeded, or IFS FAILURE if some sort
of error occurred. In the latter case, the external variable ifserr will be set to indicate the
nature of the error. Possibilities are:
• IFSE BAD HEADER if img doesnt point to a valid image.
• IFSE NOT IMAGE if there is no data associated with the header, i.e., the field
img->ifsptr is set to NULL.
• IFSE NO OPEN if the specified file cant be opened (usually meaning that the name
is invalid or that the user doesnt have write permission in the directory in which she
is trying to put the image.
• IFSE IO ERR if some sort of I/O error occurred. The standard system I/O library
variable errno may contain additional information about the nature of the error. Note
that IFSE NO OPEN is a subclass of the IFSE IO ERR error, so one can check for
both automatically by using a construct of the form if (ifserr & IFSE IO ERR) action
to take();
• IFSE BAD NAME if some error occurred while translating the name.
BUGS/NOTES:
• The function of ifspot has been superceded by the newer function ifsPutImg. Starting
with version 4 of IFS, ifspot is just a dummy routine which remaps its arguments
and calls if- sPutImg. Ifspot, however is convenient and supported
• Ifspot does not write out any tail information associated with the image.
1.4. IFS FUNCTION LISTING
1.4.36
57
ifsPrsFN
ifsPrsFN: expand a filename
NewName = ifsPrsFN(Name,rc);
char *NewName; char *Name;
int *rc;
This function is very useful for programmers writing systems like ifs. The ifs user is also
welcome to use it.
IfsPrsFN scans a string Name looking for references to environment variables or abbreviations for a users home directory of the form user such as is provided by the Unix
C-shell. It returns a pointer to the expanded name, or NULL if it failed. The space for
the expanded name is allocated using calloc, so it may be cfreeed when the user is through
with it. A status code is returned through the pointer rc. This code will be 0 if it was
successful, 1 if the expansion failed (such as by reference to an unset environment variable),
or 2 if the routine had an internal error (such as a failure in a call to calloc).
Environment variables are specified by prefixing the name with a dollar sign $. The name
of the environment variable may contain any alphanumeric character, and is terminated
by the first non-alphanumeric character found. The name may be enclosed in braces to
isolate it from other characters, such as when the user desires the first character after
the environment variable name to be an alphanumeric. Also, if the name is enclosed in
braces, almost any printable character can be part of the variable name rather than just
alphanumerics. Environment variable substitution is done on a strict left to right basis. A
reference to some users home directory may be specified in the same manner as that allowed
by the Unix C-shell. If the first character in a filename begins with a tilde character, then
the word immediately following the tilde (where word is terminated by the first character
which is not alphanumeric or underscore) is taken to be the name of some users login id;
the name of the users home directory is substituted for the login id construct. Examples
Assume the following environment variables and login ids:
$i ifs $file output $J john /usr/users/myhome john /usr/users/alpha Then the following
names expand as:
NAME myfile$i myfile.$i /myfile john/$file.$i $J/$file.$i $ibase $ibase EXPANSION
myfileifs myfile.ifs /usr/users/myhome/myfile /usr/users/alpha/output.ifs /usr/users/alpha/output.ifs no expansion unless environment variable ibase set ifsbase (braces isolate i
from base)
58
1.4.37
CHAPTER 1. THE IFS SYSTEM
ifsPutImg
ifsPutImg: open a file and write an IFS image to it
Usage:
rc = ifsPutImg(Image, FileName, Prompt, WriteTail);
int rc;
IFSHDR *Image;
char *FileName;
char *Prompt;
int WriteTail;
IfsPutImg writes an IFS image to some file FileName. If WriteTail is false (zero), then any
tail information associated with the image will not be written to the new file. IfsPutImg
returns IFS SUCCESS if all went well, or IFS FAILURE if something went wrong, in
which case the external integer variables ifserr and column can be examined to determine
the nature of the error. If FileName is NULL, then the image is written to stdout. If
FileName is a null string (that is, FileName = “”), then a filename will be read in from
stdin using the routine ifsGetFN. In this case, if Prompt isnt NULL and stdin corresponds
to a terminal, then the string Prompt will be printed on the terminal (actually, to stderr)
before reading a name. If stdin is not attached to a terminal, such as when input is being
piped in from another program, then the printing of the prompt string is suppressed. If a
filename is read in from stdin, and it is a single dash, then the image itself will be written
to stdout. Filenames are expanded using ifsPrsFN, so they may contain such things as
environment variable names and login id constructs. IfsPutImg works by opening the
specified file and then calling ifsWrImg to do the actual work of storing the image. It then
closes the file when its done (unless it wrote to stdout). Note that calling ifsPutImg with
Filename = NULL is essentially the same as calling ifsWrImg directly.
The complimentary routine to ifsPutImg is ifsGetImg.
1.4. IFS FUNCTION LISTING
1.4.38
59
ifsRdHdr
ifsRdHdr: read an IFS image header from an open file
Usage:
hdr = ifsRdHdr(file);
IFSHDR *hdr;
FILE *file;
IfsRdHdr reads an image header from a previously opened file. It does not read in any
data or tail information for the file. It returns a pointer to the new image header, or NULL
if it failed, in which case the external integer variables ifserr and column can be examined
to determine the nature of the error. Space for the header is allocated dynamically, and
may be freed (using cfree) when the user through with it.
After the header is read, the file pointer is positioned so that the next character read
from the file will the first byte of the data stored in the file. Hence, ifsRdHdr does scan
past any padding at the end of the header.
There is no complimentary routine for writing headers to open files in this version of
IFS. Writing a header to a file without writing any data would not make sense Accordingly,
there is a function ifsWrImg, but not an ifsWrHdr.
60
1.4.39
CHAPTER 1. THE IFS SYSTEM
ifsRdImg
ifsRdImg: read an IFS image from an open file
img = ifsRdImg(File,ReadTail); I
FSHDR *img;
FILE *File;
int ReadTail;
IfsRdImg reads an image from a previously opened file. If ReadTail is false (zero), then
any tail information associated with the image will not be read in. It returns a pointer to
the new image header, or NULL if it failed, in which case the external integer variables
ifserr and column can be examined to determine the nature of the error. Space for the
header, data, and tail is allocated dynamically, each may be freed (using cfree or ifsfree)
when the user is through with it.
IfsRdImg will always read the entirety of an image file (including tail informa- tion
and any padding after it), discarding the tail if it is not wanted, and the file read position
will be set so that the next read request will start with the first byte after the end of
the image. If File corresponds to a disk file, this just means the read pointer will point
to the end-of- file (unless some garbage has been concate- nated to to the of the image
file). If File does not correspond to a disk file, such as when piping is being used and File
is stdin, this means the file read pointer is positioned so that subsequent read requests
(including read, scanf, getchar, another call to ifsRdImg, etc.) will properly read new data
rather than reading padding characters left over from the end of the first image file. The
complementary routine to ifsRdImg is ifsWrImg.
1.4. IFS FUNCTION LISTING
1.4.40
61
ifssiz
ifssiz: Get size (lengths of all dimensions) of an IFS image
Usage:
dlength = ifssiz(image);
int *dlength;
IFSHDR *image;
Ifssiz is used to determine the lengths of each dimension of an IFS image. It returns a
pointer to an integer array, the various elements of which indicate the lengths of each
dimension of the image, and also how many dimensions the array defined as. The array
will have N+1 elements, where N is the number of dimensions of the image. The first
element of the array (element number zero) gives the number of dimensions for the image.
Subsequent elements of the array give the length each dimension, where the dimensions are
in order of ascending rank; i.e., element one gives the number of pixels per line (number of
columns) for the image, element two gives the number of lines (rows), element three is the
number of frames, and so forth.
The space for the array returned by ifssiz is automatically allocated using standard system calls (e.g., calloc), and as such may be released back to the system with the appropriate
calls (free, cfree) when the user is through with the array.
CAUTION – Potential Memory Leak: a popular programming error is to use ifssiz
inside a loop without a free, and then wonder why the program is growing. For this reason,
most programmers prefer ifsdimen to perform the same function.
If there is some error in ifssiz, then the external variable ifserr will be set to some error
code as defined in the file ifserr.h ( most likely IFSE BAD HEADER or IFSE NO MEM.)
Example usage:
int nrows, ncols, ndims, * dimlength;
IFSHDR * image2d; ...
make or read in image pointed to by image2d ...
dimlength = ifssiz(image2d);
ndims = dimlength[0];
if (ndims != 2) { /* Exit with nasty error messages ... */ }
ncols = dimlength[1];
nrows = dimlength[2];
cfree( (char *) dimlength );
62
1.4.41
CHAPTER 1. THE IFS SYSTEM
ifsslice
ifsslice: take a complete slice of a two-d or three-d image Usage:
newimg = ifsslice(oldimg, string, value);
where string is a char pointer pointing to a string. The following are legitimate strings
“frame”, “f”, “row”, “r”, “column”, “col”, “c”.
Passing any one of these strings will inform the function that the slice should be taken
with that particular dimension (row,col or frame) held constant at the integer parameter
value. i.e, if the string is frame and the value = 10, then a 2-d slice of the 3-d image at
frame=10 is returned. Similarly for row and col slices. This is a generic slice program
for 2-d and 3-d images. Using ifsslice on 1-d images will return with a copy of the image
pointer and a warning. Similarly, a text string of frame on a 2-d image returns a copy of
the image pointer and a warning.
Returns:
This function returns NULL if an error occurs, and returns an error code through the
external variable ifserr. External variables: ifserr, IFSSLV Special routines used: ifscfree,
ifsdie, ifswarn, ifsexwin, ifsexwin3d, ifssiz Ifsslice extracts a complete slice of a two-d or
three-d image from the constituent image. The datatype of the sliced new image is exactly
that of the old image. Note that the slice is complete in all dimensions except in one
dimension.
1.4. IFS FUNCTION LISTING
1.4.42
63
ifsversion
ifsversion: display version numbers Usage
ifsversion(file);
FILE *file;
Ifsversion will write the version numbers of all the IFSfunctions it knows about to the
specified file. Typically, file will be stdout or stderr. For each function ifsversion knows
about, it will print the name of the function, its version number, and the date it was last
modified. In rare cases there may be some additional information printed.
64
1.4.43
CHAPTER 1. THE IFS SYSTEM
ifsWrImg
ifsWrImg: write an IFS image to an open file
rc = ifsWrImg(Image,File,WriteTail);
int rc;
IFSHDR *Image;
FILE *File;
int WriteTail;
IfsWrImg writes an IFS image to some opened file File. If WriteTail is false (zero), then any
tail information associated with the image will not be written to the new file. IfsWrImg
returns IFS SUCCESS if all went well, or IFS FAILURE if something went wrong, in
which case the external integer variables ifserr and column can be examined to determine
the nature of the error. The complimentary routine to ifsWrImg is ifsRdImg.
1.5
IFS Error Codes
This section describes the various error flags which may be set (in the global variable ifserr)
when an error occurs in an IFS routine. These are defined in the #include file ifserr.h Each
error is represented by a bit or set of bits in ifserr; hence it is best to test for specific bits
rather than using a standard comparison (==). Note that all of the IFS error codes have
names which are of the form IFSE xxxxxx, where xxxxxx is the actual name for the error.
IFSE ERROR This is a combination of all possible errors. It is to be -1, i.e., all bits
of the variable ifserr are set. Hence, all other error codes are subclasses of this code. IFS
routines do not generally return this code. It generally indicates that either (a) an error was
too complex for IFS to figure out, or (b) it was such a rare error that it was not considered
important enough to define a separate code for the error. IFSE BAD HEADER or IFSE
BAD HDR The pointer you passed to a function does not correspond to a valid header for
an IFS image. Most IFS routines double-check image headers before doing anything, and
will exit with IFSE BAD HEADER set if the header is not valid.
IFSE NULL HEADER or IFSE NULL HDR The value NULL was passed to some IFS
routine where you should have passed a pointer to an image header. The most likely cause
of this is calling a routine to get or put a pixel in an image, when you havent yet created
(or read in) the image. This error is a subclass of the error IFSE BAD HEADER. IFSE
NO OPEN A file could not be opened for I/O activities. Usually this indicates that the file
does not exist (on reads), or that the user does not have access permission for the specified
file or directory. IFSE NO OPEN is a subclass of the error IFSE IO ERR.
IFSE NOT IMAGE An error occurred when attempting to read an image header from
a file. This usually means the file is too small to possibly be an IFS image. An image
header alone occupies at least 1 block, where a block is normally to be 512 bytes. IFSE
NOT IMAGE is 55 a subclass of the error IFSE IO ERR. IFSE BAD NAME A filename
1.6. IFS DATA TYPES
65
is considered invalid. This is normally set within routines such as ifsPrsFN when a name
expansion fails (such as by reference to a file fred/ file.ifs when user fred doesnt exit). IFSE
BAD NAME is a subclass of the error IFSE IO ERR.
IFSE IO ERR Some sort of error occurred while performing I/O. The system global
variable errno may contain additional information about the error. Common causes are
(a) encountering an unexpected EOF, (b) inability to write output due to a full disk or
users disk quotas exceeded, (c) inability to open a file.
IFSE BAD DTYPE The datatype (short, int, float, etc.) is invalid or unrecognized by
a particular routine. Usually this will only occur if you pass an invalid argument to an
image creation routine (e.g., ifsmkh or ifscreate). It might also occur on routines which
read or write data in images if the image header has been corrupted, or if the function is
not capable of working on an image of a particular data type (for instance, it would make
little sense to pass a complex format image to a histogram routine). IFSE BAD POS Some
coordinate (array index) is illegal for the specified operation, such as trying to access a
pixel in column 30 of an image which is only 20 columns wide. Note that the routines
which read or write single pixels currently do NOT check to see if coordinates are within
bounds. This is a flaw with IFS which will probably be fixed at a later time.
IFSE WRONG NDIM The routine called does not work with images of the dimensionality of the image being used. An example would be trying to extract a window (2D
subimage) from a 1-dimensional array.
IFSE NOT SUPPORTED The specified function is not currently allowed. Usually this
indicates a function which is not yet implemented, but which is intended to be implemented.
In rare cases it may indicate that a function is obsolete (a separate error code may later
be for this).
1.6
IFS Data Types
This section describes the various data types that IFS version 5 understands. IFS has
a certain basic set of names it recognizes for data types, and which it actually puts into
image headers; in addition, it recognizes a number of synonyms for data types which
it automatically remaps into the real data type. Some of these synonyms are machine
dependent, for instance, a type of int may map to 16bit on one machine and 32bit on another
machine. The final authority on data types and synonyms is the header file ifstypes.h which
contains a table relating synonyms to the proper data type. Note that the data types ARE
case sensitive. The possible data types are:
• 8bit Signed byte. Synonyms are byte, char, i1, and I1.
• u8bit Unsigned byte. Synonyms are ubyte, uchar, u1, and U1.
• 16bit Signed 16 bit integer. Synonyms are short, i2, and I2.
66
CHAPTER 1. THE IFS SYSTEM
• u16bit Unsigned 16 bit integer. Synonyms are ushort, u2, and U2.
• 32bit Signed 32 bit integer. Synonyms are int, long, i4, and I4.
• u32bit Unsigned 32 bit integer. Synonyms are uint, ulong, u4, and U4 (this data
type is not supported by many compilers)
• 32flt 32 bit floating point number. Synonyms are float, real, real*4, r4, and R4.
• 64flt 64 bit floating point number. Synonyms are double, real*8, r8, and R8.
• 32cmp Complex number consisting of two 32flt numbers (real and imaginary parts).
Synonyms are complex, complex*4, c4, and C4.
• 64cmp Complex number consisting of two 64flt numbers (real and imaginary parts).
Synonyms are complex*8, c8, and C8.
• struct Arbitrary user defined structure. Although IFS will read and write such images, it supplies no intrinsic routines to manipulate such images.
1.7
The structure of an IFS image
An IFS image, whether it is in a disk file or in program memory, is stored as a set of three
distinct pieces. When written to disk, each piece will begin on a block boundary, where
the size of a block is given by the constant BLOCKSIZE, which is de ned the #include file
ifs.h. Hence, there may be garbage bytes between one section and the next.
The first piece is a header for the image. This header contains all sorts of information
relevant to the processing of the image, along with information intended solely for the users
benefit. Sample items in the header include the number of dimensions the image has, how
long each dimension is, who the creator of the image is, and so on.
The second entity in an image is the actual image data. The data is just stored in one
long linear array, in exactly the same way that any C program stores arrays. The user can
directly access this data if he/she so desires, although the usual way to get at the data is
to use various IFS routines such as ifsigp and ifsipp.
The third part of the image is the tail. The tail is just a block of data at the end of the
file which IFS places no particular interpretation upon. It is up to the users programs to
manipulate and understand the contents of the tail. An sample usage for the tail might be
to store the text of a spoken message for which the data block was the digitized message.
In most cases, it is not necessary for the user to directly alter any of the information in the
image header as the IFS routines themselves will fill in the header with all the information
needed to process the image, and all of the user information fields will be set to default
values which are fine for most applications. However, at times it is desirable to alter fields
in the header, which requires that the user know what the fields in the header are, and how
1.7. THE STRUCTURE OF AN IFS IMAGE
67
they are used. The header actually consists of several C structures. These structures are
defined in the #include file ifs.h. The header actually consists of two types of structures.
The first structure is the main image header structure, and contains most of the relevant
information about the image, such as the number of dimensions, the format of the data,
etc. This structure is the so called IFSHDR structure which one refers to when one declares
an image pointer variable in a program (e.g: IFSHDR * img1, * img2;).
Along with the IFSHDR structure exists a variable number of dimension sub- headers.
There is one of these sub-headers for each dimension of the image, e.g., a 2-d image would
have two sub- headers. The main piece of information in these subheaders is how long
each dimension is. This structure goes by the name IFSDIM. The IFSDIM structures
come directly after the the main header structure, both in the in-core images and the disk
images. Hence, given a pointer to the main header structure, and the sizes of the headers,
one can easily generate pointers to any of the dimension headers. The macro ifsgetdim
(defined in ifs.h) may be used for this:
IFSDIM * dim; IFSHDR * img; dim = ifsgetdim(img,2);
will return a pointer to the third dimension sub-header (the first sub-header has number
of zero).
1.7.1
The image header fields
• char ifsmgc[4] This is the magic number field in the header. This field is used by the
various routines as a way of verifying the validity of the header passed to them. If
this field does not contain a special magic number (really, a character string rather
than a number), then the IFS routines will assume that an invalid pointer was sent
to them. The user should never alter this field.
• int ifsbb This is the number of bytes in a physical block, when images are stored on
disk. This value is set to the constant BLOCKSIZE, which is defined in ifs.h. For
all systems to date, the blocksize is 512. When images are written to disk files, the
header always starts at block 0, and the data always begins at the start of the next
block after the header, i.e, there may be a small amount of wasted space between the
end of the header and the start of the data, if the header does not completely ll the
last block it occupies.
• int ifssoh This is the block number of the first block of the header. This is always
set to zero, at least for the time being. int ifssod This is the number of the block at
which the data starts. The user can position directly to the start of the data array
by using fseek to position an I/O pointer to the ifssod*ifsbb byte of the le. Of course,
this only works for disk files.
68
CHAPTER 1. THE IFS SYSTEM
• int ifssot This is the block number of the start of the tail for the le. If this field
is negative, it indicates that there is no tail present; taking the absolute value of it
would give the block number at which the tail would be if it existed. char * ifstail
This is a pointer to the image tail, for an in-core image. If there is no tail, this is set
to NULL.
• int ifstsz This gives the size of the tail in bytes. If there is no tail, this is just zero.
char ifsfc[8] This is the file class field. This is not used by UNIX installations of IFS,
and is intended for systems running operating systems other than UNIX.
• char ifsct[8] This is the file class type field. This also is not used and is for non-UNIX
systems.
• char ifsunm[32] This field is used to store the name of the owner of the file, as a
null terminated character string. Note that since one byte must be reserved for the
terminating null, that the effective username length is 31 characters. The user can
put anything here she wants. When a user creates a new image, this field is filled in
with his/her login id.
• char ifscdt[32] This is a character string giving the time and date at which the image
was created. This is automatically filled in when a user creates a new image, but the
user can change it if she so desires. As with the name field, there can be up to 31
characters, plus the terminating null character.
• char ifscpg[32] This is a character string giving the name of the program which created
the image. When a user first creates an image, this field is normally lled with the
name of the subroutine which actually created the image (e.g.: ifscreate).
• char ifsver[8] This is a character string giving the version of the program which created
the image. E.g., V 1.00 or Ver 1A or something in that vein. Certain routines such
as ifscreate will stuff their version number in here.
• char ifsrs1[40] This is just space reserved for future expansion.
• char ifsdts[16] This is a character string giving the units of data for the pixels in the
image, e.g., for an intensity image, this field might contain lumens. One must make
sure not to use names for units which exceed 15 characters. The default for this field
is just pixels.
• float ifsdsc This field gives a scaling factor for the data in the image. This can be used
along with the data offset (defined below) to convert values in the image array to
some other scale. This might be used for example, if an image is taken and digitized
using some measuring instrument, and later it is found that the instrument was off
center (a data offset) or suffered from some sort of compression (scaling) problem.
1.7. THE STRUCTURE OF AN IFS IMAGE
69
The default for this field is 1.0. float ifsdo This field gives an offset which should
be applied to the data in an image. I.e., the real value for a point in the image
array should be calculated as realvalue = storedvalueif sdsc + if sdof Note that the
routines which get values from the image array (such as ifsfgp) do NOT apply the
scaling factors. The default for this field is 0.0.
• char ifsdt[16] This is a character string which tells what number format the pixels in
the image are stored in, such as u8bit or 32flt.
• char comp This byte denotes the type of compression that is used. Right now, the
only values are the integers one and zero, where one means use compression and zero
means no compression. A program can use this by executing img-¿comp = 0: /*
suppress compression */ before the execution of the ifs write file command.
• int dtype This is a numeric encoding of the ifsdt field which has been added to the
header structure with version four of IFS. This has been added to increase the speed
at which certain routines work.
• int ifsbpd This is the number of bytes which are needed to store a single pixel value,
i.e., its the sizeof whatever data type is used for the image. Of course, this field can
be deduced from the ifsdt field.
• int ifsdims This gives the number of dimensions for the image. This refers to the
number of indices needed to get at values in the image array, i.e, the pixels themselves
dont count as a dimension. For instance, an image which has 10 rows and 20 columns
is a 2-d image. Some other nomenclatures might refer to this as a 3-d image, where
the third axis is the pixel measurement axis (range, brightness, or whatever).
• char * ifsptr This is a pointer which gives the address of the of the first data element
in the data array, for in-memory images. When fies are written to disk, the value
NULL is written for this field. This is normally automatically set to the correct value
when an image is read in, although the user can alter it to point to some other array.
• int * ifsdln This is a pointer to an array which is used when calculating the address of
any arbitrary point of the image. This array has ifsdims elements. The first element
is just set to 1, the next element is the number of columns in the image, the next
element is the number of rows times the number of columns, the fourth element is
numcols * numrows * numframes, and so on. If the user has an N dimensional image,
and the N-length vector V gives the coordinate of some point in the image (i.e., V
= [ col, row, frame, cube... ]), then the dot-product of V and ifsdln will give you an
offset which may be added to the starting address of the image to find the desired
element, assuming the starting address is an appropriately declared pointer. If the
starting address is declared as a char * (such as with the header field ifsptr) then the
70
CHAPTER 1. THE IFS SYSTEM
offset must be scaled by the data size (ifsbpd). This may sound confusing, but really
just represents the usual way that a set of indices are converted to absolute memory
addresses for an array, whether by IFS or the C language itself. Note: the array itself
is not written to disk when an image is stored. It is created when an image is read in
(such as by ifsRdImg, ifspin, etc.) using information in the dimension sub-headers.
• char * userptr This field is not used at present.
• char ifsrs3[4] More reserved space.
1.7.2
The dimension sub-header fields
For each dimension of the image, there will be a structure of the following form tacked on
after the end of the main header structure. The user can obtain a pointer to one of these
structures using the ifsgetdim macro, or can calculate their positions manually using the
size of the main header and subheaders. The dimension sub-header fields are:
• int ifslen The length (number of elements) of this dimension.
• int ifsrnk The rank of this dimension. The rank of the dimension defines the order in
which the dimensions are actually stored in memory. The dimension with the lowest
rank is the dimension which changes most rapidly. Hence, the dimension with rank
1 is equivalent to columns, the rank 2 dimension is rows, the rank 3 dimension is
frames, and so on. Note that images are stored in row-major order (as with all C
arrays), which is contrary to the way some languages store arrays Fortran for instance
stores in column- major form. Also note that the first dimension subheader after the
main header is not necessarily the header for the lowest rank (columns) although the
IFS routines do by convention store the dimension subheaders in order of ascending
rank, this is not a requirement.
• char ifsdir[8] The direction of this dimension. This is for images for which lines are
not always stored in a top to bottom, left to right form. For instance, some camera
systems scan from left to right on one line, then go from right to left on the next
line, and store the data in the same form. This would be known as forward-backward
alter storage. Other possibilities include forward (normal),backward, and backwardforward alter. Currently, IFS does NOT recognize this field, and treats all images
as being stored in forward format. This is only for possible future expansion. The
string fwd is placed in this field.
• char ifsxun[8] This is a character string which gives the units for this dimension e.g.,
inches or mils. Make sure not to use names exceeding 7 characters. The default for
this field is pixels.
1.7. THE STRUCTURE OF AN IFS IMAGE
71
• float ifsdsc The scaling factor to apply to this dimension, analogous to the scaling
factor which exists in the main header.
• float ifsdof The scaling offset for this dimension.
• char rs4[32] Reserved space.
72
CHAPTER 1. THE IFS SYSTEM
Chapter 2
The FLIP library
FLIP Floating Point Image Processing Library This library assumes input in the
form of a floating point ifs image. Outputs will be floating point images as well. The
library is designed to optimize computation speed, while providing the programmer with
easy-to-use tools. If all the inputs are not float, the function returns -1. If the
input dimensions are not compatible, the function returns -2. Note that no
error messages are printed, so the user must check the value returned.
NOTE: to build programs which use many of the flip functions, you will need the math
library. Thus, build using the -lm switch which invokes the math library.
You can call many of the FLIP functions from the command line. See section 2.6.
2.1
Example Program Using the FLIP functions
In this section, we suppose that the user must write a program which computes an expression using derivatives and transcendantal functions. That is, at every pixel, compute
r = exp(−
(fx )2 + (fy )2
)
2τ 2
(2.1)
where fx denotes the partial derivative with respect to x and τ is a constant.
#include <ifs.h>
#include <flip.h>
int main(int argc, char *argv[])
{
IFSIMG inimg,outimg,temp1,temp2;
int len[3];
inimg=ifspin(argv[1]); // read the file specified by the first argument
len = ifssiz(inimg)
; // get the sizes of the dimensions
outimg=ifscreate("float",len,IFS_CR_ALL,0);
73
74
CHAPTER 2. THE FLIP LIBRARY
temp1=ifscreate("float",len,IFS_CR_ALL,0);
temp2=ifscreate("float",len,IFS_CR_ALL,0);
temp3=ifscreate("float",len,IFS_CR_ALL,0);
fldx(inimg,temp1);
flmultv(temp1,temp1,temp1); // square the value
fldy(inimg,temp2);
flmultv(temp2,temp2,temp2); // square the value
fladdv(temp1,temp2,temp3); // add them
fldivs(temp3,temp3,-(2.0 * tau)); // divide
flexp(temp3,outimg);
ifspot(outimg,argv[2]);
free(ifssiz);
ifsfree(outimg,iFS_FR_ALL); // doesn’t need to be done since the program is exiting here
// but this is good practice
ifsfree(inimg,iFS_FR_ALL);
ifsfree(temp1,iFS_FR_ALL);
ifsfree(temp2,iFS_FR_ALL);
ifsfree(temp3,iFS_FR_ALL);
} // end of this program
In the example above, note the use of in-place operations. with multv, divs, and
exp. Calls with the same input and output and NOT allowed in derivative or other
neighborhood-based operations.
2.2
Matrix Operations
This section contains just a few commonly used matrix operations. These are developed
using the Numerical Recipies in C (Cambridge University Press) conventions and utility
functions. These functions are defined in flip.h.
It is recommended that users set up their vectors to start at 1, not zero, to be consistent
with math. For example, use v=vector(1,n);
2.2.1
dmatrix
Create a matrix of doubles
usage:
double **r;
r = dmatrix(1,6,1,8);
Will create a matrix of doubles, rows indexed from 1 to 6, columns indexed from 1 to 8.
2.2. MATRIX OPERATIONS
2.2.2
matrix
Create a matrix of floats
usage:
float **r;
r = matrix(1,6,1,8);
Will create a matrix of floats, rows indexed from 1 to 6, columns indexed from 1 to 8.
2.2.3
dvector
Create a vector of n doubles
usage:
double *r;
r = dvector(1,6);
Will create a vector of 6 doubles, lower index is 1, higher index is 6.
2.2.4
vector
Create a vector of n floats
usage:
float *r;
r = vector(1,6);
Will create a vector of 6 floats, lower index is 1, higher index is 6.
2.2.5
ivector
Create a vector of n integers
usage:
int *r;
r = ivector(1,6);
Will create a vector of 6 integers, lower index is 1, higher index is 6.
75
76
2.2.6
CHAPTER 2. THE FLIP LIBRARY
free ivector
Free a previously created vector of integers.
Usage:
free_ivector(v,1,6)
2.2.7
free dvector
Free a previously created vector of doubles.
Usage:
free_dvector(v,1,6)
2.2.8
free vector
Free a previously created vector of floats.
Usage:
free_vector(v,1,6)
2.2.9
transpose
Usage:
void transpose(double **v,int rows,int cols,double **vtrans);
The matrix v, which has rows rows and cols columns, will be transposed and stored in
matrix vtrans. The user must ensure that the size of vtrans is correct. This program
assumes the indexing starts with 1, as is the common convention.
2.2.10
ifsmatmult
Multiply two matrices
Usage:
int ifsmatmult(double **, double **, double **, int,int,int,int);
ifsmatmult(x, y, z, xrows,xcols,yrows,ycols);
For the two matrices to be compatible, xcols must equal yrows. The program will return
an error if this is not true. The resultant matrix, z, must be of size xrows x ycols. The
matrices are assumed to have been created with the matrix function defined above, with
a lower index of 1.
2.3. ARITHMETIC OPERATIONS ON IMAGES
2.2.11
77
ifsinverse
double ifsinverse(double **,double **, int);
Computes the inverse of a matrix, using numerical recipies conventions
Usage:
det = ifsinverse(double **x, double **xinv, int n);
x is an nxn matrix. After the call, xinv will contain the inverse of x. The function will
return the determinant.
2.2.12
jacobi
Computes eigenvalues and eigenvectors. call:
float **a,**v;
float *d;
int n;
a = matrix(1,n,1,n); // input float matrix a must be symmetric
v = matrix(1,n,1,n); // output matrix v will contain eigenvectors as columns
d = vector(1,d);
// output vector d will contain eigenvalues
...
jacobi(a,n,d,v,&nrot); // computes everything
// nrot will contain the number of rotations required
2.3
2.3.1
Arithmetic Operations on Images
flabsolute
Take absolute value of an image
int flabsolute(inimage,outimage}
2.3.2
fladds
Add a scalar to an image
int fladds(IFSIMG img1, IFSIMG img2, float scalar)
78
2.3.3
CHAPTER 2. THE FLIP LIBRARY
fladdv
Add two images
int fladdv(IFSIMG img1, IFSIMG img2, IFSIMG img3)
The first input image is img1, the second input image is img2, the output is img3.
Corresponding pixels of img1 and img2 are added and the result stored in img3.
2.3.4
flclip
Limit permitted brightness in images
int flclip(IFSIMG img1, IFSIMG img2,float scalar)
The first image, img1 is clipped so that its maximum value is scalar.
2.3.5
flcp
Copy an image
flcp(IFSIMG inimg, IFSIMG outimg)
The image inimg will be copied to the image outimg
2.3.6
fldivs
Divide an image by a scalar.
fldivs(IFSIMG img1 IFSIMG,img2,float scalar)
Each pixel of img1 will be divided by the scalar, and the result stored in the corresponding
pixel if img.
2.3.7
fldivv
Divide two images
int fldivv(IFSIMG img1, IFSIMG img2, IFSIMG img3);
Each pixel of img1 is divided by the corresponding pixel of img2, and the result stored in
img3. Note, no check for divide by zero is performed, so the special floating point value
INFINITY can occur.
2.3. ARITHMETIC OPERATIONS ON IMAGES
2.3.8
79
flexp
Exponentiate
int flexp(IFSIMG img1, IFSIMG img2)
Each pixel of img2 is independently exponentiated. Result stored in img2.
2.3.9
flln
Compute logs of pixels independently
int flln(IFSIMG img1, IFSIMG img2)
Computes out[i] = log(in[i]);
2.3.10
flmults
Scalar multiply
int flmults(IFSIMG img1, IFSIMG img2,float scalar)
Multiply each pixel of img1 by scalar and store the result in the corresponding pixel of
img2.
2.3.11
flmultv
Vector multiply
int flmultv(IFSIMG img1, IFSIMG img2, IFSIMG img3)
Each pixel of img1 is multiplied by the corresponding pixel of img2, the result stored in
img3. This is equivalent to the MATLAB .* operation.
2.3.12
flneg
Negate image pixels
int flneg(IFSIMG img1, IFSIMG img2)
Each pixel of img1 is multiplied by -1 and the result stored in img2.
2.3.13
flnorm
int flnorm(IFSIMG img1,float *norm)
Returns the 2-norm of the image img1. The second argument is a POINTER to a float,
and the norm (square root of sum of squares of pixels) will be returned.
80
2.3.14
CHAPTER 2. THE FLIP LIBRARY
flrec
Reciprocal of an image
int flrec(IFSIMG img1, IFSIMG img2)
Each pixel of the input image, img1, is divided into one, and the result stored in the
corresponding pixel of img2. This function tests for divide by zero, reports an error, and
exits early.
2.3.15
flsq
Square pixel values
int flsq(IFSIMG img1, IFSIMG img2)
For every pixel, out[i] = in[i] * in[i] ;
2.3.16
flsqrt
Square root of pixels
int flsqrt (IFSIMG img1, IFSIMG img2)
The square root of each pixel it taken. If any pixels are negative, the function will return
FAILURE (0), else it will return SUCCESS (non-zero).
2.3.17
flsubs
Scalar subtract. Subtract a scalar from every element of an image
int flsubs(IFSIMG img1, IFSIMG img2,float scalar)
2.3.18
flsubv
int flsubv(IFSIMG img1, IFSIMG img2, IFSIMG img3);
Vector subtract Each pixel in the second image, img2 is subtracted from the corresponding
pixel in img1, and the result stored in img3. Valid for any number of dimensions.
2.3.19
flthresh
Threshold
int flthresh(IFSIMG img1, IFSIMG img2,float scalar,float bkgnd)
Each pixel of input image img1 is tested against scalar. If pixel > scalar, the pixel is
unchanged, otherwise, the corresponding pixel in the output image is set to the value
bkgnd. Valid for any number of dimensions.
2.4. IMAGE MANIPULATION FUNCTIONS
2.3.20
81
flthresh2
Threshold
int flthresh2(IFSIMG img1, IFSIMG img2,float scalar,float bkgnd,float fgnd)
Each pixel of input image img1 is tested against scalar. If pixel > scalar, the pixel is set
to fgnd, otherwise, the corresponding pixel in the output image is set to the value bkgnd.
Valid for any number of dimensions.
2.4
2.4.1
Image Manipulation Functions
flone border
int flone_border(IFSIMG img1, IFSIMG img2)
Sets the left-most and right-most columns of the image to one. Then set the top row
and bottom row to one. Result copied into img2.
2.4.2
flpad
Image pad int flpad(IFSIMG img1, IFSIMG img2)
Input image img1 is padded by taking the leftmost pixel on each row and replacing it by
the next- to-leftmost. Same thing on right side, top, and bottom. If 3D, appropriate things
are done to the first and last frame.
2.4.3
flplanar
Image pad using linear interpolation.
int flplanar(IFSIMG img1, IFSIMG img2)
The leftmost pixel on each line is replaced by what it would be if the first three pixels were
lin- early increasing in brightness – the zeroth pixel is the linear interpolation of the first
and second. Similar on right side, top, and bottom. Only implemented for 2D images.
2.4.4
flrotate
Rotate an ifs image about its center.
int flrotate(IFSIMG input,IFSIMG output,float angle);
82
CHAPTER 2. THE FLIP LIBRARY
Unlike most other flip functions, flrotate will accept input of any data type. It runs faster
with unsigned byte and float. It will also rotate 3D IFS images, but all frames will be
rotated by the same amount. (Thus a color image can be rotated). Returns: zero if it runs
successfully.
Returns: -2 if the input and output images have different dimensions.
2.4.5
flshx
Shift image in the x direction
int flshx(IFSIMG img1,int dx, IFSIMG img2)
input image img1 is shifted by dx pixels in the x direction and the result stored in img2.
Operation is valid for 3D images.
2.4.6
flshxy
Shift image
int flshxy( IFSIMG int img1,dx,int dy, IFSIMG img2)
Input image img1 is shifted dx pixels in x and dy pixels in y. Operation is valid for 3D
images.
2.4.7
flshy
Shift image
int flshy(IFSIMG img1,int dy, IFSIMG img2)
Shifts image in y direction. Operation is valid for 3D images.
2.4.8
flzero border
Set border pixels to zero.
int flzero_border(IFSIMG img1, IFSIMG img2)
Valid for two dimensional images.
2.5.
2.5
DERIVATIVE OPERATORS
83
Derivative Operators
In computing image derivatives, (with the exception of flDoG), DO NOT CALL THESE
FUNCTIONS IN-PLACE. That is, never do something like
fldx(img,img);
// bad bad bad
Performing an image derivative in-place will radically distort the output. Always use
different destination and source image. Because flDoG actually creates a temporary image,
it is OK to use it in-place.
Except for DoG (derivative of a Gaussian), all the derivative operators listed below
support 1-D (signals) 2-D (images) and 3-D (volumes) data sets. In all cases, x denotes
the column and y the row.
flDoG only supports two-dimensional images, however, if the problem is 2D, it probably
does a better job in the presence of noise.
2.5.1
flDoG
Computes the derivative of an isotropic, zero mean Gaussian, and applies it to an image.
This is by far the best derivative program in the flip library.
int flDoG(IFSIMG a, IFSIMG b, float sigma, int order, int direction);
The ifs image a will be differentiated and its derivative stored as image b. By choosing
combinations of the arguments, order and direction, one can compute the blurs and derivatives in the table below. Sigma is the standard deviation of the Gaussian.. The equation
of the Gaussian is
2
1
x + y2
exp −
2πσ
2σ 2
The kernel used will be computed by the program by extrapolating out to greater than
3 sigma. This could result in a large kernel and long run times. You may get a warning
message to this effect, but the program will run anyway. Because the kernels may get large,
the program checks that the image is at least large enough to hold the image. If not, the
program will exit with an error message.
The message “Something wrong with arguments to function ifsDoG” indicates that an
impossible combination of order and direction were specified.
84
CHAPTER 2. THE FLIP LIBRARY
Order
0
1
2
3
0
1
2
3
0
1
2
3
2.5.2
Direction
0
0
0
0
1
1
1
1
2
2
2
2
Function
Convolution with zeroth order derivative of a Gaussian, which is a blur
First deriv wrt x (column direction)
Second deriv wrt x (column direction)
Third deriv wrt x (column direction)
Blur, same as 0 0
First derivative wrt y (row direction)
Second derivative wrt y (row direction)
Third derivative wrt y (row direction)
Blur, same as 0 0
Undefined. Will produce an error message
Cross second deriv (Partial**2 f )/ (Partial x Partial y)
Undefined. Will produce an error message
flGabor
Performs a Gabor filtering on an image.
The Gabor filter, applied to a point xy in an image, computes
02
x + γ 2 y 02
x0
g(x, y; σ, θ, λ, γ, ψ) = exp −
cos
2π
+
ψ
2σ 2
λ
Here, the Gabor is implemented by using the ifs function flGabor The normal to the
line has the angle theta, not the line itself where
x0 = x cos θ + y sin θ
(2.2)
0
(2.3)
y = −x sin θ + y cos θ
The parameters involved in the construction of a 2D Gabor filter are:
σ , The standard deviation of the Gaussian function
θ , The orientation of the normal to the parallel stripes of the Gabor function
λ , The wavelength of the sinusoidal function
γ , The aspect ratio of the exponential.
ψ , The phase of the sine wave.
Note that flGabor gives the user the options of selecting ψ, the phase of the cosine, which
allows the user to determine whether to be sensitive to lines, rising edges, or falling edges.
A phase of zero means use a cosine centered at zero, which is best for finding lines (two
opposite edges), One may detect very narrow lines by using a ψ of zero and a narrow
2.5.
DERIVATIVE OPERATORS
85
wavelength. A ψ of −π/2 changes the cosine to a sine which makes it sensitive to finding
positive edges. A ψ of π/2 changes the cosine to a downward sine which makes it sensitive
to finding falling edges.
USAGE:
int flGabor(IFSIMG input,IFSIMG output,float sigma,float theta,
float lambda, float gamma,float psi);
The input and output images must both be float; They must be the same size.
sigma is the standard deviation of the blurring Gaussian of the Gabor filter.
theta is the orientation of the Gabor filter. It is an angle, measured in radians, relative to
the x axis. If theta =0 (for example), the filter will prefer vertical edges. Lambda is the
wavelength of the sinusoid. Good example settings for finding lines (not edges) are lambda
= 4, psi = 0;
2.5.3
fldx
Estimate partial deriviative in the x direction. Do not use in-place.
int fldx(img1,img2);
IFSIMG img1,img2;
Compute an approximation of the first deriviative with respect to x by the difference
between pixels in an image images. For each pixel, outrow[j][i] = (inrow[j][i+1] - inrow[j][i1])*.5
If all the inputs are not float, the function returns -1. If the input dimensions are not
compatible, the function returns -2. Note that no error messages are printed, so the user
must check the value returned.
2.5.4
fldx back
Backward difference. Do not use in-place.
The forward and backward differences both estimate the first partial derivative with respect
to x, but do it with higher resolution (and higher noise sensitivity) than fldx. They are
useful in iterative algorithms where you do one iteration using the forward difference, and
the next using the backward.
int fldx_back(IFSIMG img1, IFSIMG img2)
Compute the backward difference between pixels in an image images. For each pixel,
outrow[j][i] = inrow[j][i+1] - inrow[j][i]; is computed. Do not use in-place. If all the inputs
are not float, the function returns -1. If the input dimensions are not compatible, the
function returns -2. Note that no error messages are printed, so the user must check the
value returned.
86
2.5.5
CHAPTER 2. THE FLIP LIBRARY
fldx forw
Forward difference. Do not use in-place.
int fldx_forw(IFSIMG img1, IFSIMG img2)
Compute the forward difference between pixels in an image images. For each pixel, outrow[j][i] = inrow[j][i] - inrow[j][i-1]; Do not use in-place.
2.5.6
fldxx
Second partial derivative with respect to x.
int fldxx(IFSIMG img1, IFSIMG img2)
Uses a 3x1 kernel. Do not use in-place. If all the inputs are not float, the function returns
-1. If the input dimensions are not compatible, the function returns -2. Note that no error
messages are printed, so the user must check the value returned.
2.5.7
fldxy
Second partial derivative with respect to x and y.
int fldxy(IFSIMG img1, IFSIMG img2)
Uses a 3x3 neighborhood. Do not use in-place. If all the inputs are not float, the function
returns -1. If the input dimensions are not compatible, the function returns -2. Note that
no error messages are printed, so the user must check the value returned.
2.5.8
fldxz
Second partial wrt x,z. Do not use in-place.
int fldxz(IFSIMG img1, IFSIMG img2)
2.5.9
fldy
First partial derivative with respect to y. Do not use in-place.
int fldy(IFSIMG img1, IFSIMG img2)
If all the inputs are not float, the function returns -1. If the input dimensions are not
compatible, the function returns -2. Note that no error messages are printed, so the user
must check the value returned
2.5.
DERIVATIVE OPERATORS
2.5.10
87
fldy back
Backward difference The forward and backward differences both estimate the first partial
derivative with respect to y, but do it with higher resolution (and higher noise sensitivity)
than fldy. They are useful in iterative algorithms where you do one iteration using the
forward difference, and the next using the backward. Do not use in-place.
int fldy_back(IFSIMG img1, IFSIMG img2)
Compute the backward difference between pixels in an image images. For each pixel,
outrow[j][i] = (inrow[j+1][i] - inrow[j-1][i])*.5; is computed. Do not use in-place. If all the
inputs are not float, the function returns -1. If the input dimensions are not compatible,
the function returns -2. Note that no error messages are printed, so the user must check
the value returned.
2.5.11
fldy forw
Forward difference
int fldy_forw(img1,img2)
IFSIMG img1,img2;
Compute the forward difference between pixels in an image images. For each pixel, outrow[j][i] = inrow[j][i] - inrow[j-1][i]; Do not use in-place. If all the inputs are not float, the
function returns -1. If the input dimensions are not compatible, the function returns -2.
Note that no error messages are printed, so the user must check the value returned.
2.5.12
fldyy
Second partial derivative with respect to y. Do not use in-place. If all the inputs are not
float, the function returns -1. If the input dimensions are not compatible, the function
returns -2. Note that no error messages are printed, so the user must check the value
returned.
int fldyy(img1,img2);
IFSIMG img1,img2;
Uses a 3x1 kernel.
88
2.5.13
CHAPTER 2. THE FLIP LIBRARY
fldyz
Second partial derivative with respect to y and z. Do not use in-place.
int fldyz(img1,img2)
IFSIMG img1,img2;
Uses a 3x3 kernel. Obviously, only meaningful for 3D data. If all the inputs are not float,
the function returns -1. If the input dimensions are not compatible, the function returns
-2. Note that no error messages are printed, so the user must check the value returned,
2.5.14
fldz
First partial derivative with respect to z. Obviously, only meaningful for 3D data. Do not
use in-place.
int fldz(img1,img2)
IFSIMG img1,img2;
If all the inputs are not float, the function returns -1. If the input dimensions are not
compatible, the function returns -2. Note that no error messages are printed, so the user
must check the value returned.
2.5.15
fldzz
Second partial derivative with respect to z. Only meaningful for 3D data. Do not use
in-place. If all the inputs are not float, the function returns -1. If the input dimensions are
not compatible, the function returns -2. Note that no error messages are printed, so the
user must check the value returned.
int fldzz(img1,img2)
IFSIMG img1,img2;
Uses a 3x1 kernel.
2.6
Running the FLIP functions from the command line
Sometimes, users want to run a FLIP function on an image without bothering to write a
program to call that function. Some programs have been written which allows the user to
do just that. These are listed below
2.6. RUNNING THE FLIP FUNCTIONS FROM THE COMMAND LINE
2.6.1
89
flpg1: single input FLIP functions
This program allows to multiple flip library functions from the command line.
Switches:
-i inputfilename
-o outputfilename
-f functionname
clip - clips the image
divs - divides by a scalar
exp
- takes the exponential of each pixel of the image
log
- takes the natural log of each pixel of the image independently
mults - takes the exponential of each pixel of the image
neg
- Negate image pixels
rec
- Reciprocal of an image
sq
- Square pixel values
curv - Curvature at every point
sqrt - Square root of pixels
subs - Scalar subtract. Subtract a scalar from every element of an image
-s optional, function-specific parameter
Example:
flpg1 -i footprint.ifs -o footdarker.ifs -f subs -s 2.0
2.6.2
der1: take a derivative of an image
This program allows to multiple flip library functions from the command line.
Switches:
-i inputfilename
-o outputfilename
-f functionname
fldx
- Estimate partial derivative in the x direction
fldx_back - Backward difference
fldx_forw - Forward difference
fldxx
- Second partial derivative with respect to x
fldxy
- Second partial derivative with respect to x and y
fldxz
- Second partial derivative with respect to x and z
fldy
- Estimate partial derivative in the x direction
fldy_back - Backward difference
fldy_forw - Forward difference
fldyy
- Second partial derivative with respect to y
fldyz
- Second partial derivative with respect to y and z
90
CHAPTER 2. THE FLIP LIBRARY
fldz
fldzz
2.6.3
- Estimate partial derivative in the z direction
- Second partial derivative with respect to z
ifsDoG:Apply a derivative of Gaussian kernel to an ifs image
usage: ifsDoG [switches]
switchs are
-i infilenname, where infilename is an ifs image
default: the null string
-o outfilenname, where outfilename is an ifs image
default: the null string
-s sigma, std. dev of the filtering Gaussian
default: sqrt of 3.0
-O
order of the derivative
zero is blur with the Gaussian,
one is first deriv, etc, up to 3rd deriv
-d
direction. This flag is the letter c for col, r for row, b for both
default: col
b option only valid for second deriv
example: ifsDoG -i myfile.ifs -o outfile.ifs -O 0 -d r
2.7
2.7.1
Image Manipulation
ifsadd
ifsadd - add two ifs images, point by point out(i, j) = in1(i, j) + in2(i, j) Usage:
int ifsadd (in1, in2, out) IFSHDR *in1,*in2,*out;
RETURNS 0 if successful, -1 if all three arguments do not have same dimensions -2 if data
type unsupported (complex double) -3 if one input has type complex and output is real
-4 if both inputs are real and output is complex CAUTION: if output is type char, values
greater than 255 will be clipped to lie between 0 and 255. NOTES: if one image is real
and the other complex, the output must be complex and the real parts of the images will
be added.
2.7.2
ifsany2any
Convert any ifs image to any data type.
Usage:
2.7. IMAGE MANIPULATION
status=ifsany2any(inputimage,outputimage);
** FUNCTION: convert data type of images, point by point
** CALL: ifsany2any (in1,out)
**
where each argument is an ifs image
** RETURNS: 0 if successful,
**
-1 if both arguments do not have the same dimensions
**
-2 if input data type unsupported (complex)
**
-3 if one input has type complex and output is real
**
-4 if input is real and output is complex
**
-5 if a zero occurs in the input
91
92
CHAPTER 2. THE FLIP LIBRARY
Chapter 3
Image Processing Subroutines
In this chapter, a number of subroutines are presented which are of general applicability.
Most have been written using pointers and sophisticated code in order to optimize speed.
3.1
Geometric functions
The following subroutines are available in the library /usr/local/libiptools.a
3.1.1
chainHull 2D
Finds the convex hull of a set of 2D points. The set is an array of structures of type Point.
Where Point is defined by typedef struct float x, float y Point; Usage:
count = chainHull_2d(Point *S, int n, Point *H);
count is an integer, the number of points in the convex hull (cannot be more than n)
n in the number of points in the input data se// S is the array of input points// H will be
filled in by the subroutine, the points in the convex hull
This software was copied from freeware:
http://softsurfer.com/Archive/algorithm 0109/algorithm 0109.htm#chainHull 2D%28%29
Note: the x,y pairs must be sorted on x, smallest first. If two pairs have the same x coordinate, then those need to be sorted by y. You can use the Unix soft program if that is
convenient, or any of a large selection of sort subroutines.
3.1.2
compute curvature
Given a set of x,y pairs, compute the curvature at every point. Usage:
compute_curvature(struct point_xy parray[],int grid_spacing,
int num_elems, double curvature[]);
93
94
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
The structure points is simply an x,y, pair defined as follows:
struct point_xy {
int x,
int y
};
The user should declare an array of these structures. num elems is the number of elements
in this array. grid spacing is the pixel width; almost always set to 1. This would be
changed only if the pixels are not square. curvature[] is an array of curvatures found by
this program.
3.1.3
InsideTriangle
Determine if a point is inside a triangle
Usage:
int InsideTriangle(int row, int col, double **X);
X is an array of doubles created with
X=dmatrix(1, 2, 1,3);
The three columns of X refer to the coordinates of the 3 vertices of the triangle. The 1,1
entry in the matrix is the row coordinate of the first vertex, The 1,1 entry in the matrix is
the column coordinate of the first vertex
3.1.4
InterpolateTriangle
Given three points in an input image, and a corresponding 3 points in an output image,
the second triangle will be deformed and filled in to match the first. Usage:
status = InterpolateTriangle (inimg, outimg, X,U,IG);
IFSIMG inimg, outimg;
double **X, **G;
int IG
X and U are matrices created using the function dmatrix. IG is a switch which, if nonzero,
will result in sub-pixel interpolation. (At the time of this writing, this feature is not
implemented.) The matrix manipulation functions are in the flip library. See section 2.2.2.
If the input image is three-dimensional and has three frames, it will be treated as color.
Three dimensional images are interpolated only in a 2D sense.
Simple Example:
3.1. GEOMETRIC FUNCTIONS
95
double **X, **Y;
X=dmatrix(1,2,1,3); // X is a matrix with rows numbered
//from 1 to 2 and columns numbered from 1 to 3.
U=dmatrix(1,2,1,3);
X[1][1] =0; X[2][1] = 0; // one corner of undistorted image
U[1][1] =0; U[2][1] =0; // this point is not moved
X[1][2] =0; X[2][2] = 10; // second corner of triangle, ten units to right of previous
U[1][2] =0; U[2][2] =10; // this point is not distorted
X[1][3] =10; X[2][3] = 5; // third corner of triangle, ten units down
//and 5 units over from first
U[1][3] =15; U[2][3] =10; // this point is moved, stretching
// the triangle up and to the right
InterpolateTriange(inputimge, outputimage,X,U,0);
96
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
Real Example
#include <ifs.h>
#include <ifsmatrix.h>
int main(int argc, char *argv[])
{
IFSIMG inimg;
IFSIMG outimg;
int row,col,nr,nc;
int len[4];
double **X, **U;
X=dmatrix(1,2,1,3);
U=dmatrix(1,2,1,3);
inimg = ifspin(argv[1]);
len[2]=nr = ifsdimen(inimg,1);
len[1]=nc = ifsdimen(inimg,0);
len[0]=inimg->ifsdims; // two dimensional
if(len[0]== 3) len[3]=ifsdimen(inimg,2);
outimg = ifscreate(inimg->ifsdt,len,IFS_CR_ALL,0);
{
float ulr,ulc,urr,urc,llr,llc,lrr,lrc,mr,mc;
ulr =0;ulc =0;
urc=nc-1;urr=0;
llr=nr-1;llc =0;
lrr=nr-1;lrc=nc-1;
mr = nr/2; mc =0;
X[1][1] = ulr;X[2][1] =ulc;
X[1][2] = urr;X[2][2] =urc;
X[1][3] = mr;X[2][3] =mc;
U[1][1] = ulr;U[2][1] =ulc;
U[1][2] = urr;U[2][2] =urc;
U[1][3] = 90;U[2][3] =mc;
InterpolateTriangle(inimg,outimg,X,U,0);
X[1][1] = urr;X[2][1] =urc;
X[1][2] = mr;X[2][2] =mc;
X[1][3] = lrr;X[2][3] =lrc;
U[1][1] = urr;U[2][1] =urc;
U[1][2] = 90;U[2][2] =mc;
U[1][3] = lrr;U[2][3] =lrc;
3.1. GEOMETRIC FUNCTIONS
97
InterpolateTriangle(inimg,outimg,X,U,0);
X[1][1] = mr;X[2][1] =mc;
X[1][2] = lrr;X[2][2] =lrc;
X[1][3] = llr;X[2][3] =llc;
U[1][1] = 90;U[2][1] =mc;
U[1][2] = lrr;U[2][2] =lrc;
U[1][3] = llr;U[2][3] =llc;
InterpolateTriangle(inimg,outimg,X,U,0);
}
ifspot(outimg,argv[2]);
}
Example makefile using InterpolateTriangle
distort1: distort1.c
cc -o distort1 distort1.c -I/Users/wes/src/ifs/MacX64/hdr \
/Users/wes/src/ifs/MacX64/ifslib/libiptools.a \
/Users/wes/src/ifs/MacX64/ifslib/libifs.a \
/Users/wes/src/ifs/MacX64/ifslib/libflip.a
3.1.5
cubic splines
IFS library:
libiptools.a
usage:
void cubic_int(int n,int m,float* f, float* t, float* fn, float* tn)
n is the number of points in the input arrays (f and t),
m is the number of points in the output arrays (fn and tn)
both the time arrays, t and tn must be initialized
example initialization
// i n i t i a l i z e t
for ( i =0; i <N; i ++) {
f [ i ]= s i n ( 2 ∗ 3 . 1 4 ∗ i / ( f l o a t ) (N− 1 ) ) ;
t [ i ]=( f l o a t ) i ;
}
// i n i t i a l i z e t n
s t e p s i z e =(N−1)/( f l o a t ) (M−1);
for ( i =0; i <M; i ++) {
tn [ i ]= t [ 0 ] + i ∗ s t e p s i z e ;
98
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
fn [ i ]=0.0;
}
∗/
void c u b i c i n t ( int n , int m, f l o a t ∗ f , f l o a t ∗ t , f l o a t ∗ fn , f l o a t ∗ tn )
3.1.6
Resampling Curves
IFS library:
libiptools.a
Resample a line or curve to ensure uniform (arc-length) sampling. Accepts a set of
points, x-y pairs, and produces a set of x-y pairs. There are two functions which must be
called to use this feature: resampleinit and resamplesub. Usage:
#include <resample.h>
struct rs * resampleinit(int m,int n);
resampleinit sets up a structure in which to hold information about the resampled data.
m is the number of points in the input image (the number of rows in the array). n is the
number of points into which to divide the resampled output image. Once initialized, the
resample function is
void resamplesub(int m, int n, double **x, double **y,struct rs *p)
The first two arguments are the same as above: m is the current number of points, n will
be the number of points on the resampled curve. x and y are matrices, x2,m , y2,n holding
the actual data. These are arrays created by resampleinit using dmatrix (see section 2.2.1).
Their indices run from 1 to 2 (for the first index) and 1 to m or n. That is, they are created
using
x = dmatrix(1,2,1,m);y = dmatrix(1,2,1,n);
.
Note that while these functions are in libiptools, the matrix manipulation functions are
in libflip, so you will need to link with both those libraries.
Example resample
In the example below is shown how to set up and call the two resample functions. It does
not include details about how to read and write the ASCII input and output files.
#include <math.h>
#include <stdio.h>
#include <ifsmatrix.h>
3.1. GEOMETRIC FUNCTIONS
99
#include <resample.h>
#define EPSILON .0001
int main(int argc, char *argv[])
{
double **x; /* the input shape, it has m points */
double **y; /* the output shape, it will have n points, uniformly spaced in arc length */
/* the shapes above are pairs of floats. The even numbered ones are x coords, the odd y
*/
int m;
/* m is the number of points in the input shape */
int n;
/* n is the number of points we demand the output shape will have */
struct rs *rsdata; // this is the structure that will persist between resampleinit
// and resamplesub
double median;
FILE *fp, *fopen();
int j;
int getshape(char *, double **, double *, double); // function to read the input file
int getnumberofpoints(char *);
void clparse(int,char *[],struct clp *);// function to parse command line, not provided here
void resamplesub(int ,int,double **,double **,struct rs *);
struct rs *resampleinit(int,int);
/* call the read subroutine to read in the x,y pairs of points */
/* we pass the address of the shape pointer to getshape because the malloc
will be done inside getshape */
m = getnumberofpoints(infile); // read the input file to get the m
n = NUMBEROFOUTPUTPOINTS;
x = (double **)dmatrix(1,m+1,1,2);
y = (double **)dmatrix(1,m+1,1,2);
m = getshape(infile,x,&median,(double) 0.0); // read the input file
rsdata = resampleinit(m,n);
resamplesub(m,n,x,y,rsdata);
//inititalize the struct
/* write out the y array */
/* actually write the x array, since we have already swaplped pointers */
fp = fopen(outfile,"w");
for(j = 1; j <=NUMBEROFOUTPUTPOINTS; j++)
100
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
{
fprintf(fp,"%lf %lf\n",x[j][1],x[j][2]);
}
fclose(fp);
return 0;
}
3.1.7
Operations on FIFO’s
The following operations are useful in manipulating first-in-first-out memories, or queues.
All of them use a structure which contains information about the queue. The structure
has the following form:
struct myque{
int ∗q ;
int f r o n t ;
int r e a r ;
int s t a r t ;
int end ;
int count ;
};
fifo init; initialize the queue
call:
void fifo_init(int D,struct myque *queue);
fifo add; add a new number to the queue
call:
int fifo_add(int item ,struct myque *queue);
fifo remove
Remove a number from the queue and return it. call:
int fifo_remove(struct myque *queue);
fifo empty
Return true (nonzero) or false (zero) if the queue is or is not empty
call:
int fifo_empty(struct myque *queue);
3.1. GEOMETRIC FUNCTIONS
3.1.8
101
MinimumRegion
This function finds and marks all minimum regions in an image.
Every pixel in the output image either is a zero or has a distinct label
A minimum region is all the same brightness and has no neighbors darker than itself.
This function only runs on Unsigned Char images
This function is an almost direct translation from algorithm 4.6 of Roerding and Meijster.
Usage:
int MinimumRegion(IFSIMG img, IFSIMG labelimg)
Note: the source code for MinimumRegion is part of the source module Watershed.c
3.1.9
Watersheds
A watershed is the boundary between two basins.
int ifs_WaterShed(IFSIMG im,IFSIMG om);
Both images must be ifs images of type int.
The output image is a segmented version of the input image. It will contain uniquely
labeled regions. WaterShed points will have brightness zero. Note that although watershed
pixels are in principal the boundaries of basins, they are not necessarliy a single pixel thick.
The WaterShed program (see section ??) adds additional features by erasing watershed
pixels and assigning them to regions. The WaterShed program also provides the user with
the ability to pseudocolor the output image.
3.1.10
ifs ccl
IFS library:
libiptools.a
This function performs connected components on an image. It is the function called
by the program named ccl. The program ccl simply reads the command line, fills in a
structure from the command line, allocates some memory, and calls this function.
The declaration for the call is
void ifs_ccl(IFSIMG, IFSIMG, IFSIMG,struct PAR *parptr);
Because there are many options, the structure is rather long. The process of defining
the structure and filling it in is given below. The user is advised to simply cut and
paste the block below. This block includes the call to ifs ccl, passing the images and
the structure. The connect type field allows the user to specify which type of connivity is
used to determine when two pixels/voxels are considered neighbors. In three dimensions,
the three types of connectivity are shown in Figure 3.1. In two dimensions, FACE is
4-connected, EDGE is 8-connected.
102
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
Figure 3.1: Three types of connectivity may be used by the connected components function
in 3D images. In 2D images, FACE is equivalent to 4-connected, and EDGE is equivalent
to 8-connected
3.1. GEOMETRIC FUNCTIONS
103
{ /∗ s t a r t o f t h e b l o c k ∗/
// t h i s i s t h e d e f i n i t i o n o f t h e s t r u c t u r e
struct PAR {
char InFileName [ 1 0 0 ] ; // not needed i f c a l l e d as a f u n c t i o n
char OutFileName [ 1 0 0 ] ; // not needed i f c a l l e d as a f u n c t i o n
float lower limit ;
// low l i m i t o f p o i n t s c o n s i d e r e d , u s u a l l y z e r o
float upper limit ;
/∗ upper l i m i t o f p o i n t s c o n s i d r e d , u s u a l l y 255 ∗/
int
connect type ;
/∗ EDGE( 2 ) ,FACE( 1 ) or VERTEX( 4 ) ∗/
f l o a t c o n n t h r e s h o l d ; // c o n n e c t i o n t h r e s h o l d . two p i x e l s w i t h
// g r e a t e r d i f f e r e n c e w i l l not be c o n n e c t e d
int
roi set ;
// s e t t o one f o r Region o f I n t e r e s t
int
roi seed [ 3 ] ;
/∗ r o i s e e d l o c a t i o n , 3 numbers ∗/
int
r o i l a b e l ; /∗ l a b e l f o r r e g i o n o f i n t e r e s t ∗/
int
roi seg out ;
/∗ segment o u t t h e r o i components ∗/
int
r o i s e g v a l u e ; /∗ v o x e l v a l u e s f o r r e g i o n o f i n t e r e s t ∗/
int
l o w e r b g l a b e l ; /∗ l a b e l used f o r l o w e r b a c k g r o u n d ∗/
int
u p p e r b g l a b e l ; /∗ l a b e l used f o r upper b a c k g r o u n d ∗/
int
m i n l a b e l ; /∗ minimum l a b e l used ∗/
int
m a x l a b e l ; /∗ maximum l a b e l used ∗/
int
c a m s i z e ; /∗ s i z e o f c o n t e n t a d d r e s s a b l e memory a r r a y ∗/
unsigned int
∗cam ;
/∗ p o i n t e r t o t h e cam ; ∗/
/∗ t h e f o l l o w i n g 3 numbers c o u l d be o b t a i n e d from t h e image
b u t a r e i n c l u d e d i n t h e s t r u c t u r e o n l y f o r e f f i c i e n c y o f a c c e s s ∗/
int nr ; // number o f rows
int nc ; // number o f columns
int n f ; // number o f frames
};
struct PAR mypar , ∗ p a r p t r ; // an i n s t a n c e o f t h e s t r u c t u r e and a p o i n t e r t o i t
void i f s c c l ( IFSIMG , IFSIMG , IFSIMG , struct PAR ∗ ) ;
unsigned int ∗ cam ptr ;
int i ;
par=&mypar ; // i t i a l i z e t h e p o i n t e r
par−>InFileName [ 0 ] = 0 ; // not needed i f c a l l e d as a f u n c t i o n
par−>OutFileName [ 0 ] = 0 ; // not needed i f c a l l e d as a f u n c t i o n
par−>l o w e r l i m i t =0;
// low l i m i t o f p o i n t s c o n s i d e r e d , u s u a l l y z e r o
/∗ upper l i m i t o f p o i n t s c o n s i d r e d , u s u a l l y 255 ∗/
par−>u p p e r l i m i t =1;
par−>c o n n e c t t y p e =1;
/∗ EDGE( 2 ) ,FACE( 1 ) or VERTEX( 4 ) ∗/
par−>c o n n t h r e s h o l d =0; // c o n n e c t i o n t h r e s h o l d . two p i x e l s w i t h
par−>r o i s e t =0;
// s e t t o one f o r Region o f I n t e r e s t
104
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
par−>r o i s e e d [ 0 ] = 0 ;
/∗ r o i s e e d l o c a t i o n , 3 numbers ∗/
par−>r o i l a b e l =0; /∗ l a b e l f o r r e g i o n o f i n t e r e s t ∗/
par−>r o i s e g o u t =0;
/∗ segment o u t t h e r o i components ∗/
par−>r o i s e g v a l u e =1; /∗ v o x e l v a l u e s f o r r e g i o n o f i n t e r e s t ∗/
par−>l o w e r b g l a b e l =1; /∗ l a b e l used f o r l o w e r b a c k g r o u n d ∗/
par−>u p p e r b g l a b e l =500; /∗ l a b e l used f o r upper b a c k g r o u n d ∗/
par−>m i n l a b e l =1; /∗ minimum l a b e l used ∗/
par−>m a x l a b e l =5000; /∗ maximum l a b e l used ∗/
par−>c a m s i z e =64000; /∗ s i z e o f c o n t e n t a d d r e s s a b l e memory a r r a y ∗/
par−>cam=0;
/∗ p o i n t e r t o t h e cam i n i t i a l i z e d l a t e r ∗/
/∗ t h e f o l l o w i n g 3 numbers c o u l d be o b t a i n e d from t h e image
b u t a r e i n c l u d e d i n t h e s t r u c t u r e o n l y f o r e f f i c i e n c y o f a c c e s s ∗/
par−>nr=i f s d i m e n ( inimg , 1 ) ; // number o f rows
par−>nc=i f s d i m e n ( inimg , 0 ) ; ; // number o f columns
par−>n f =1; // number o f frames
/∗ now t h a t t h e s t r u c t u r e i s i n i t i a l i z e d , we a r e r e a d y t o c a l l t h e
c o n n e c t e d components f u n c t i o n ∗/
/∗ i n i t i a l i z e t h e c o n t e n t −a d d r e s s a b e memory∗/
/∗−−−−a l l o c a t e CAM and c a r d i n a l i t y a r r a y s ∗/
par−>cam = ( unsigned int ∗ ) c a l l o c ( par−>c a m s i z e , s i z e o f ( int ) ) ;
i f ( par−>cam == NULL) {
p r i n t f ( ” E r r o r : Unable t o a l l o c a t e %d e l e m e n t CAM\n” , par−>c a m s i z e ) ;
e x i t ( −1);
}
/∗−−−− i n i t i a l i z e CAM ∗/
cam ptr = par−>cam ;
for ( i = 0 ; i < par−>c a m s i z e ; i ++) ∗ cam ptr++ = i ;
/∗ n o t i c e c c l t a k e s t h r e e image i n p u t s , however , t h e t h i r d one
i s o n l y t o u c h e d i f ROI p r o c e s s i n g i s r e q u e s t e d , so we use outimg ∗/
/∗ FINALLY, we c a l l i t ! ∗/
i f s c c l ( inputimg , i m a g e a f t e r c c l , outimgwithROI , p a r p t r ) ;
}
3.1. GEOMETRIC FUNCTIONS
3.1.11
105
ifscfft2d
IFS library:
libiptools.a
ifscfft2d - perform in-place 2D fast Fourier transform Usage:
len = cfft2d(img_ptr, type) IFSHDR * imgptr; int type;
Ifscfftt2d performs an in-place 2-D fast Fourier transform on a complex ifs image. The
transform is performed in place on 8BYTE-PER-PIXEL (complex float) data only! Note
that ffts only work on images of dimension 2n×2n . The second argument is an indicator for
forward or inverse fft, -1 for forward, +1 for inverse. If there is some error, the subroutine
exits to the user with an error message. Possible errors are:
• Image dimensions are not a power of two
• Image data type is not complex float
106
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
3.1.12
ifsc2imag
IFS library:
libiptools.a
ifsc2imag - extract imaginary part of a complex ifs image, point by point Usage:
val = ifsc2imag (in1, out) int val; IMSHDR *in1,*out;
RETURNS 0 if successful, -1 if both arguments do not have same dimensions -2 if data
type unsupported (complex double) CAUTION: if output is type char, values greater than
255 will be clipped to lie between 0 and 255.
3.1. GEOMETRIC FUNCTIONS
3.1.13
107
ifsc2mag
IFS library:
libiptools.a
ifsc2mag - return magnitude of a complex ifs image, point by point Usage:
val = ifsc2mag (in1, out) int val; IMSHDR * in1,*out;
RETURNS 0 if successful, -1 if both arguments do not have same dimensions -2 if data
type unsupported (complex double) CAUTION: if output is type char, values greater than
255 will be truncated to 255.
108
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
3.1.14
ifsc2phase
IFS library:
libiptools.a
ifsc2phase - return phase of a complex ifs image, point by point Usage:
val = ifsc2phase (in1,out) int val; IMSHDR *in1,*out;
RETURNS 0 if successful, -1 if both arguments do not have same dimensions, -2 if data
type unsupported (complex double) CAUTION: if output is type char, values greater than
255 will be truncated to 255.
3.1. GEOMETRIC FUNCTIONS
3.1.15
109
ifsc2real
IFS library:
libiptools.a
ifsc2real - return real part of a complex ifs image, point by point Usage:
val = ifsc2real (in1,out) int val; IMSHDR *in1,*out;
RETURNS 0 if successful, -1 if both arguments do not have same dimensions -2 if data
type unsupported (complex double) CAUTION: if output is type char, values greater than
255 will be truncated to 255.
110
3.1.16
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
ifsInsert2Dinto3D
IFS library:
libiptools.a
ifsInsert2Dinto3D - inserts a 2D image as a single frame into a 3D ifs image
Usage:
val = ifsInsert2Dinto3D (IFSIMG in,IFSIMG out,int frame)
RETURNS 0 if successful, -1 if both arguments do not have same dimensions or if data
type unsupported (complex double) CAUTION: if output is type char, values greater than
255 will be converted to char.
3.1. GEOMETRIC FUNCTIONS
3.1.17
111
ifsmult
IFS library:
libiptools.a
ifsmult - multiply two ifs images, point by point Usage:
int ifsmult (in1, in2,out) IFSHDR *in1,*in 2,*out;
RETURNS 0 if successful, -1 if all three arguments do not have same dimensions -2 if data
type unsupported (complex double) -3 if one input has type complex and output is real
-4 if both inputs are real and output is complex CAUTION: if output is type char, values
greater than 255 will be truncated to 255. NOTES: if one image is real and the other
complex, the output must be complex and the real parts of the images will be added.
112
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
3.1.18
ifsrecip
IFS library:
libiptools.a
ifsrecip take reciprocal of an ifs image, point by point Usage:
int ifsrecip (in1,out) IFSHDR *in1,*out;
RETURNS 0 if successful, -1 if both arguments do not have same dimensions -2 if data
type unsupported (complex double) -3 if one input has type complex and output is real
-4 if both inputs are real and output is complex CAUTION: if output is type char, values
greater than 255 will be truncated to 255. NOTES: if one image is real and the other
complex, the output must be complex and the real parts of the images will be added.
3.1. GEOMETRIC FUNCTIONS
3.1.19
113
ifssub
IFS library:
libiptools.a
ifssub subtracts two ifs images, point by point. The second argument is subtracted
from first. Usage:
int ifssub (in1, in2, out) IFSHDR *in1,*in2,*out;
RETURNS 0 if successful, -1 if all three arguments do not have same dimensions -2 if data
type unsupported (complex double) -3 if one input has type complex and output is real
-4 if both inputs are real and output is complex CAUTION: if output is type char, values
greater than 255 will be truncated to 255. NOTES: if one image is real and the other
complex, the output must be complex and the real parts of the images will be added.
114
3.1.20
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
ifsvidscale
IFS library:
libiptools.a
ifsvidscale converts an ifs image, of any data type, to an unsigned char ifs image. The
brightness values are scaled to lie between 0 and 255. Usage:
int ifsvidscale (IFSHDR in1, IFSIMG out,float *minvalptr,float *maxvalptr, int switch3d)
The two arguments minvalptr and maxvalptr are pointers to floats, and will be returned
by ifsvidscale, reporting the minimum and maximum of the input images, respectively.
The switch3d argument, if not ==0, indicates that the frames of the image should be
scaled independently. Note that in this case, the minvalptr and maxvalptr apply only to
the last frame in the sequence.
RETURNS 0 if successful
The function will scale minimum brightness - maximum brightness to 0 to 255.
NOTES: The second image must be unsigned char.
Example:
img2=ifscreate("u8bit",len,IFS_CR_ALL,0);
ifsvidscale(inimage, img2,&mymin,&mymax,0);
3.2
3.2.1
Functions to modify images
ifscolormap
IFS library:
libiptools.a
The function ifscolormap is similar to the program ColorMap in that it allows you to assign
a number of different colors to an 8-bit, unsigned image. That is, each of the possible 256
brightness levels will be assigned a color. There are many ways to assign an R, G, and B
value to a single pixel brightness. The function offers several different ways to interpret
the brightnesses.
Usage:
void ifscolormap( IFSIMG inputimage, IFSIMG outputimage, int
X);
Where inputimage is an unsigned 8 bit fs image, and X is an integer between 0 and 6 which
tells the program which color map to use as follows:
0 grey scale
1 inverse gray scale
2 bronson
3 hot metal
4 heated spectrum
5 log
6 random
3.2. FUNCTIONS TO MODIFY IMAGES
115
The input image must be unsigned 8 bit, since there are only 256 unique colors .
The input image must be two-dimensional.
The output file must be a 3 D unsigned 8 bit ifs image, with 3 frames, with the usual
interpretation that frame 0 is red, frame 1 is green, and frame 2 is blue. The function does
not create this image, a pointer to it is passed by the calling function.
example use
int l e n [ 4 ] ;
IFSIMG inimage , outimg ;
extern void i f s c o l o r m a p ( IFSIMG , IFSIMG , int ) ;
...
len [0]=3;
l e n [ 1 ] = i f s d i m e n ( inimage , 0 ) ; // g e t number o f c o l s
l e n [ 2 ] = i f s d i m e n ( inimage , 1 ) ; // g e t number o f rows
l e n [ 3 ] = 3 ; // image w i l l have 3 frames
outimg=i f s c r e a t e ( ” u 8 b i t ” , l e n , IFS CR ALL , 0 ) ;
i f s c o l o r m a p ( inimg , outimg , 3 ) ; // use a h o t m e t a l c o l o r map
...
i fs C Vp ot ( outimg , ”myimage . png” ) ;
...
3.2.2
Lower Complete Images
To understand this, first recall that in a digital image, a maximum is a connected set
of pixels, which are all the same brightness, with the property that no pixel in this set
is adjacent to any brighter pixel. A minimum is a similarly defined set. A plateau is
a connected set of pixels, which are all the same brightness, but which has at least one
neighboring pixel which is brighter and one which is darker. Plateaus cause probems to
some algorithms.
A lower complete image is one in which every pixel is either a maximum, a minimum,
or has at least one lower neighbor. That is, a lower complete image has no plateaus. This
function provides that capability.
void Build_Lower_Complete(IFSIMG inimage,IFSIMG LowerCompleteImage)
The input, inimage, must be unsigned 8 bit. The output, LowerCompleteImage, must be
type int. The brightness values will be changed to ensure the image is lowercomplete.
3.2.3
zoomupdown
Will zoom an image
Example usage:
116
CHAPTER 3. IMAGE PROCESSING SUBROUTINES
img1 = ifspin("/Users/wes/Images/asterix256.ifs");
len[0]=2;
len[1]= ifsdimen(img1,0) * 2;
len[2] =ifsdimen(img1,0) * 1;
img2 = ifscreate(img1->ifsdt,len,IFS_CR_ALL,0);
zoomupdown(img1,img2);
The input image will be zoomed to the size of the output image. The output image will
be overwritten with the zoomed version of the input image. The two images do not have
to have proportional dimensions. For example, inputimage could be 512 x 512 and output
image could be 640 x 480. Of course, unless the images have proportional dimensions, the
output image will be distorted.
3.3
3.3.1
Random number generators
gaussrand
IFS library:
libiptools.a
double gaussrand()
returns a Gaussian-destributed random number with standard deviation of one and mean
of zero.
Chapter 4
Image Synthesis Programs
4.1
Qsyn-synthesize range images
Qsyn generates synthetic altitude images of objects which are composed of quadric surfaces
or pieces of quadric surfaces. Usage:
qsyn [-f][-l labelfile.ifs] [-o depthfile.ifs] formatfile.q
The -f switch is optional and if used will cause the output file to be floating point.
The -l switch is optional and if used will cause qsyn to also output a label image, in which
each surface is a different brightness – the image is segmented.
If the -o switch is not used to provide a name for the altitude output image, then after
running, the program will ask for the name of the output altitude image.
The label image is an image in which every pixel belonging to the same region is labeled
the same. Thus, the label image is equivalent to the output of an ideal segmenter. This
can be used to test the performance of segmentation algorithms.
Qsyn is a program which generates synthetic altitude images of objects which are
composed of quadric surfaces or pieces of quadric surfaces. The image data is in an unsigned
byte format, although a few minor changes could be made to Qsyn to allow for some other
output data type. Image manipulation is done using the IFS image manipulation routines
in use at Communication Unlimited. Qsyn generates altitude images, i.e., two dimensional
images which contain three dimensional information, where the coordinates of a pixel (its
row and column index) correspond directly to the x and y values for the pixel, and the
pixel value itself (the datum) corresponds directly to the altitude or z value for the point.
Hence, a point in three-space at position [x, y, z] corresponds to some pixel in an image I:
[x, y, z] → I[r, c]
(4.1)
where I[r,c] is the value of the pixel at row r, column c of the image. r, c, and I[r,c] are
linearly related to y, x, and z, respectively. In Qsyn, the linear relationship is simply taken
117
118
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
to be r = y, c = x, and I[r, c] = z. Note that an altitude image is not the same as a
range image, which is also commonly used to represent three-dimensional images. In a
range image, the pixel value represents the distance from some point in three space to a
fixed reference position (i.e., the view point), whereas an altitude image is based on the
distance to some reference plane, hence a range image is actually a perspective projection
of an altitude image.
Qsyn generates images composed of quadric surfaces. An image may contain any number of surfaces; in addition, each surface may also have constraints placed on it. These constraints are also quadric surfaces (quadric inequalities). A quadric surface is a 3-dimensional
surface which may be described by a general quadric equation:
q(x, y, z) = Ax2 + By2 + Cz2 + F yz + Gxz + Hxy + P x + Qy + Rz + K = 0
(4.2)
This includes common shapes such as spheres, cones, and planes. Qsyn works by generating
a quadric surface which is bounded by a set of quadric constraints. For example, an image
of an open-ended can maybe produced by synthesizing an image of a cylinder which is
constrained by two planes perpendicular to the cylinder. The constraints would specify
those points on the cylindrical surface which lie above the lower plane and below the upper
plane. As a matter of terminology, I will use the term quadric section to refer to a quadric
surface along with a set of constraints on that surface. In order to use Qsyn, you must
understand the coordinate systems it uses to orient surfaces and sections. Theoretically,
you could place surfaces wherever you wanted by specifying the appropriate coefficients in
the quadric equation 4.2. In practice, this is a pain since the coefficients are a function of
the objects position and orientation as well as its shape. E.g.,
x2 + y 2 + z 2 = r 2
(4.3)
describes a sphere of radius r centered at the origin. The quadric coefficients are A = B =
C = 1; K = −(r2 ). Moving this sphere so that its center is at location (x, y, z) = (10, 5, 0)
gives:
(x − 10)2 + (y − 5)2 + z 2 = r2
(4.4)
which when put into the form of equation 4.5 looks like
x2 + y 2 + z 2 − 20x − 10y + (125 − r2 ) = 0.
(4.5)
Rotating an object affects the quadric coefficients in a still more complicated way. Qsyn
allows you to specify a surface using any coordinate system you desire; you may then
translate or rotate the object to move it to a different coordinate system.
Qsyn uses several different coordinate systems to ease the task of creating images which
are composed of multiple surfaces. Figure 4.1 shows the various coordinate systems used,
and are described here. Each surface (including constraints) is defined in terms of its own
local coordinate system. Typically you would choose the coordinate system in which it was
4.1. QSYN-SYNTHESIZE RANGE IMAGES
119
easiest to describe the shape you want. Each quadric section has its own coordinate system,
known as a section or object coordinate system. The latter term is perhaps misleading in
that what you think of as an object may actually consist of more than one section.
By specifying the relationship between the local coordinate system for each surface in a
section to the sections coordinate system, you specify how the various parts of the section
fit together. This essentially is used to relate the constraints to the actual surface being
synthesized, with the coordinate system of the surface itself (its local system) typically
being coincident with the section coordinate system. This is not a requirement though;
the surface and its constraints are all placed relative to a common section system, rather
than the constraints being placed relative to the local system of the surface.
The next higher level coordinate system is the reference or base coordinate system. This
is the base coordinate system for the entire image to be synthesized. Its relationship to the
object coordinate systems is the same as that of the object to local coordinate systems.
By specifying the relative locations of each object coordinate system in the base system,
you define how the various quadric sections fit together, and define what the over all image
will look like.
The highest level coordinate system is the viewpoint coordinate system. This corresponds to the coordinate system for the image array, and hence, the display equipment.
The x and y axes correspond to the horizontal and vertical axes of the display, and the z
axis is the actual pixel value, i.e., the brightness or color of the pixel indicates the altitude.
The relation between the view coordinate system and the base coordinate system specifies the position which you are actually seeing the image from. In many cases the two
coordinate systems may be coincident, or one merely a translation of the other.
Note that the term “viewpoint system” is somewhat misleading in that this is not a
range image; this transform really defines a plane of projection for the image (which is
fully described at the level of the base system). The projection plane itself is the xy plane
of the viewpoint coordinate system. The image array itself is just a finite piece of this
infinite projection plane. Specifically, the origin [x, y] = [0, 0] corresponds with the pixel at
location [col, row] = [0, 0] in the image array (which is in one corner of the image). Hence,
if you define the objects in your image to lie around the origin (in the base coordinate
system), when you display the image you will probably find that all of the objects will
lie in one corner of the image, unless you have displaced the base system relative to the
viewpoint system. Put simply, the origin of the base system will be in one corner of the
image you synthesize unless you make sure to move it – and you may end up not seeing
parts of your objects since they will be clipped at the image borders.
In Qsyn, the relations between coordinate systems are expressed in terms of six basic
motions: translations along the three coordinate axes, and rotations about the axes. Motions along the axes go by the names of movex, movey, and movez. Rotations about the
axes go by the names of rotx, roty, and rotz, or alternatively as yaw, pitch, and roll, respectively. When specifying the relationship between two coordinate systems, the motions
are given in terms of the higher coordinate system, i.e., the higher level system is the base
120
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
Figure 4.1: Coordinate systems used by Qsyn
4.1. QSYN-SYNTHESIZE RANGE IMAGES
121
Figure 4.2: Order of motions
system. For example, to specify that the local coordinate system for a constraint has its
origin at location (x, y, z) = (10, 5, 2) in the section coordinate system, you would specify
the motion as the motion as (movex 10, movey 5, movez 2), as this would be specifying the
origin of the higher system (the section system) in terms of the lower (the local system).
This is easy to see for motions which are pure translations, but may provide a source of
confusion when rotations are involved.
The best way to regard the motions is as being object oriented. Although you are
specifying the relationship between two coordinate systems, the lower coordinate system
can be regarded as an object in the higher coordinate system (imagine that the lower level
system has a cube sitting at its origin, and you are moving the cube around in the higher
level system). Initially, the two coordinate systems start out with coincident axes, i.e., the
lower system is an object sitting at location (0, 0, 0) in the higher system. If you specify a
motion of movex10, then all objects in the higher level system are displaced 10 units down
the x axis. If you specify a motion of roll20, then all objects swivel around the z axis of
the higher system. This is not the same as simply rotating the lower coordinate systems
coordinate system by 20 degrees! If the lower systems origin coincides with the higher
systems origin, then the effects are the same; however, if the origin points do not coincide,
then the lower systems origin will be seen to swing on an arc around the higher systems
axis. Hence, the rotation will also cause a translation in two of the axes of the higher
122
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
system. Figure 4.1 illustrates this, and also show how the order of motions is important.
Qsyn works by reading a file which describes the image to be synthesized. This file contains the quadric coefficients for each surface and constraints (in their own local coordinate
system), and movement commands which specify how the coordinate systems are located.
Qsyn reads this specification file and generates the image; when it is done, it prompts
for the name of a file to write the image to. The output file is a 2-d IFS format image.
The data type for the image will be unsigned byte or float (depending on a command line
switch), with the value of a pixel indicating its height. The nearer a pixel is, the higher
its pixel value. The format for the image specification file is relatively simple. Presented
below is a sample specification file; the format is described below. It is a text file composed
of several blocks. These blocks may contain other blocks within them.
At the beginning of the file is a header block which specifies the size (number of rows
and columns) of the image to be generated, the number of quadric sections to synthesize,
and a set of motions which will translate the reference coordinate system to the viewpoint
coordinate system.
The motions which specify the relationship between the two coordinate systems are
actually an instance of a type of sub-block known as a coordinate transform block, or just
transform. These transforms occur in several places, and always have the same format.
There are six basic motions which may be specified in a transform block. There are also
several complex motions which are simply composites of the basic motions. All of the
motions are specified by some keyword describing the motion to perform, followed by the
parameters appropriate for that keyword. The six basic motions are those mentioned
earlier: movex, movey, movez, rotz(roll), roty(pitch), and rotx(yaw). The movex, movey,
and movez commands can also be shortened to x, y, and z. Each motion takes a single
argument which is the amount to move or the angle to rotate (in degrees). To specify a
complete transform block, you merely specify an arbitrary number of basic motions (in
arbitrary order), followed by the keyword end. The basic motions are performed in the
order specified. A sample transform block might look like:
movex 20 movey 10 pitch 30 end
This will shift an object 20 units along the x axis, 10 units in y, and swing the object 30
degrees around the y axis.
The composite motions are just shortcuts for specifying certain common sets of motions.
The combination ‘movex 10 movey 20 movez 30’ can be specified more rapidly as ‘movexyz
10 20 30’ or just ‘xyz 10 20 3’. Similarly, rpy 10 20 30 is short for roll 10 pitch 20 yaw 30. All
six basic motions can be expressed using the commands rpyt θz , θy , θx , ∆x , ∆y , ∆z and trpy
θz , θy , θx , ∆x , ∆y , ∆z . Rpyt does the rotations first, then the translations; trpy performs
the translations first. Note that the syntax for trpy is inconsistent in that although the
translations are performed first, they are the last arguments specified for the command.
After the header block there comes a set of section blocks, one block for each quadric
section to synthesize. Each block describes a surface to generate, and all the constraints
4.2. 3DSYN-SYNTHESIZE DENSITY IMAGES
123
for the surface. The quadric section blocks are in turn composed of smaller blocks. The
first sub-block is a transform block which converts from the section coordinate system
to the base coordinate system. This is followed by a surface block. A surface block
contains a set of quadric coefficients to describe a single quadric surface, and also contains
a transform block which translates the surfaces local coordinate system to the section
coordinate system. After the surface block comes the number of constraints, followed by
a set of constraint blocks. A constraint block is identical to a surface block except that it
contains a flag indicating which way the constraint inequality goes (e.g., choose all points
in the surface below some plane ).
4.2
3dsyn-synthesize density images
CONVERSION TO LATEX NOT FINISHED
3dsyn generates synthetic three dimensional density images of objects which are composed of quadric surfaces or pieces of quadric surfaces. Usage:
3dsyn format file.q
3dsyn is a program which generates synthetic three dimensional density images of objects
which are composed of quadric surfaces or pieces of quadric surfaces. The image data is
in an unsigned byte format, although a few minor changes could be made to 3dsyn to
allow for some other output data type. Image manipulation is done using the IFS image
manipulation routines in use at Communication Unlimited.
3dsyn generates density images, i.e., three dimensional images which contain three
dimensional information, where the coordinates of a voxel (its row and column and frame
index) correspond directly to the x, y, and z values for the voxel, and the voxel value itself
(the datum) corresponds directly to the density for the point. Hence, a point in three-space
at position [x, y, z] corresponds to some voxel in an image I:
[x, y, z] → [f, r, c]
(4.6)
where I[f, r, c] is the value of the voxel at frame f, row r, column c of the image. The
variables, f, r, c, and I[f, r, c] are linearly related to z, y, and x, respectively. In 3dsyn, the
linear relationship is simply taken to be f = z, r = y, and c = x, and I[f,r,c] = density.
3dsyn generates images composed of quadric surfaces. An image may contain any
number of surfaces; in addition, each surface may also have constraints placed on it.
These constraints are also quadric surfaces (quadric inequalities). A quadric surface is
a 3-dimensional surface which may be described by a general quadric equation:
q(x, y, z) = Ax2 + By 2 + Cz 2 + F yz + Gxz + Hxy + P x + Qy + Rz + K = 0.
(4.7)
This includes common shapes such as spheres, cones, and planes. 3dsyn works by generating a quadric surface which is bounded by a set of quadric constraints. For example, an
124
#
#
#
#
#
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
This is a Qsyn description file which will generate a simple altitude image
of a clipped lead sticking through a hole in the under side of a printed
circuit board. The lead will have a "clinch angle" (angle that lead is
bent away from pointing straight up) of 80 degrees, and a "lead angle"
(angle in the xy plane -- the plane representing the PC board) of 30 degrees.
#The image is composed of 4 pieces (quadric sections):
# 1. A cylindrical piece which is the body of the lead.
# 2.A plane with a hole in it which represents the circuit board.
# The cylinder in (1) goes through the hole.
# 3.A sphere at one end of the cylinder in (1) [same radius as the
# cylinder] which terminates the lower end of the lead.
# 4.A second spherical section which caps the other end of the lead.
#
# This second sphere has a larger radius than the sphere, so that
#
# the higher end of the cylinder is clipped almost in a plane.
#
# Note that if this cap wasnt here, youd be able to see inside the #
# cylinder in (1)! #
# #
# NOTE: This file is not directly suitable to be passed as input to Qsyn.
#
# Qsyn understands comments in a file (comments going from
#
# the # symbol to the end of the line). If you insist on removing them, a little #
# cleverness under Unix systems will make it suitable: #
# #
# sed s/#.*$//example.q | Qsyn #
# #
# The strange looking "sed" command will edit out the comments before #
# piping the file into Qsyn. #
# itself. #
# # #####################################################################
Figure 4.3: Qsyn expample page 1
4.2. 3DSYN-SYNTHESIZE DENSITY IMAGES
125
#-------------------------Header section------------------------------------128 128 #Dimensions of image
4 # Number of quadric sections in image
# This is the ’base’ to ’viewpoint’ coordinate system transform block:
roll 030 #0 # Rotate 30 degrees about the Z axis
# This rotates EVERYTHING by 30 degrees, and is
# what gives the lead a "lead angle" of 30 degrees.
# Note that Im really rotating the board too, but
# since the board is an infinite xy-plane, you cant
# tell it. Hence, a more technically accurate, but
# slightly more cumbersome way to do this is to have
# the "roll 30" in the section to base transforms
# of each of the actual pieces making up the lead.
movex 64 # Now Ill do a translation in X and Y so that
movey 64 # the object is centered in the image rather
movez 75 # than in the lower left corner.
end # movez 75 is just cause I want board at Z = 75. \
#
#----------------------End of header section--------------------------------# This is the first quadric section. It describes the plane which represents
# the printed circuit board.
# SECTION to BASE coordinate system transform block.
# All of the surfaces in this sections (that is, including the constraint )
end # A Null block (i.e., the two systems here
#
are coincident).
####### Define quadric surface:
# The quadric surface coefficients:
000
000
001
0 # The plane "Z = 0"
# and the transform relating the LOCAL system to the SECTION system:
end # Null block. Systems are coincident
#######end of definition for the quadric surface.
Figure 4.4: Qsyn example , page2
126
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
#
Now: define the constraints on the above surface:
1 # this is the number of constraints.
#
1 1 1 # constraint quadric coefficients
0 0 0
# X*X+Y*Y+Z*Z=100, i.e., a sphere of
0 0 0 -100 # radius 10.
#
#LOCAL to SECTION transform for this constraint:
end
#Once again, no transform.
# And lastly, a "<" or ">" symbol which indicates on which side of the
# constraint surface the object must lie:
>
# The ">" specifies that my constraint
#
equation actually is
#
X*X + Y*Y + Z*Z > =100
# Hence, only points on the plane Z = 0 which
# are OUTSIDE this sphere are valid.
# This puts a hole at the center of my plane.
#
#
#
#
#
#
#
#
This is the second quadric section. Its the cylinder which makes up the
body of the lead. I originally define the cylinder as lying on the Z axis
(which also makes it easy to specify constraints on it). Then I use the
SECTION to BASE transform to tip the cylinder (and its constraints) over
to give the appearance of a clinched lead.
The SECTION to BASE transform: lead is clinched 80 degrees. The "movez 6"
displaces the cylinder upwards 6 units, which is needed because otherwise
the lead will be embedded in the board plane, rather than lying just above it
pitch 080 movez 6 end
# Define my cylinder: cylinder on Z axis with radius 6:
110 000 000 -36
end # No transform.
Figure 4.5: Qsyn example, page3
4.2. 3DSYN-SYNTHESIZE DENSITY IMAGES
127
# Now, specify my constraints. Remember that the cylinder is defined as
# lying on the Z axis (even in the SECTION system, since the above transform
# was null), so the constraints clip the cylinder at right angles to its
# axis. Now, the SECTION to BASE transform applies to both the cylinder
\# and its constraints, so the net effect is to pitch the CLIPPED cylinder
# by 80 degrees.
#
2 # there will be two constraints.
0 0 0 0 0 0 0 0 1 0 end > # Constraint 1: Z > =0
0 0 0 0 0 0 0 0 1-34 end<
#Constraint 2: Z<=34
#
# So, now I have a cylindrical piece lying between Z = 0 and Z = 34.
# Note that the cylinder is NOT capped at the ends.
# Section 3. This is a sphere which will cap off the lower end of the lead.
movez 6 end # The "movez 6" is same as in section 2.
1 1 1
0 0 0
# Sphere of radius 6 (at the origin in the local
0 0 0
# system)...
-36
end
#... and at the origin in the SECTION system...
#
but centered at x, y, z = 0,0,6 in the BASE system.
0
# Zero constraints.
Figure 4.6: Qsyn example, page4
128
#
#
#
#
#
#
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
Lastly, section 4, the sphere capping off the upper end of the lead.
You need to pay careful attention to the movements for this piece to
observe how the spherical patch does indeed end up capping off the
cylinder described in section 2. Note that the movements given here
certainly do not describe the ONLY valid way to get the patch; any of
a variety of movements would do the trick.
pitch 080 movez 6 end
1 1 1
# sphere of radius 8.
0 0 0
0 0 0
-64
movez 28. 7085 end
# That goofy number you have to work out from
# the geometry of the situation. The sphere
# is placed so that it will sit right at the
# end of the clipped cylinder. The sphere
# will intersect the cylinder at a height of
# Z = 34 (or Z = 40 after the "movez 6" is
# applied).
1 # 1 constraint:
0 0 0 0 0 0 0 0 1 -34 end > #only want that part of the
# sphere which will cap off
# the cylinder; I dont want
# to put a big ball at the
# end of the cylinder.
#### and now, the name of the file to write the image to:
pcb_lead.ifs
Figure 4.7: Sample Qsyn input file, page5
4.2. 3DSYN-SYNTHESIZE DENSITY IMAGES
129
128 128
2
end
end
000
000
1 0 -1 -20
end
4
000
000
1 0 0 -20 end >
000
000
1 0 0 -30 end <
000
000
0 1 0 -20 end >
000
000
0 1 0 -30 end <
end
000
000
-2 0 -1 +70
end
4
000
000
1 0 0 -30 end >
000
000
1 0 0 -35 end <
000
000
0 1 0 -20 end >
THIS IS NOT COMPLETE
Figure 4.8: Another example Qsyn description: a plane at z = 0 with two sloped planes in
a rectangular area between x = 20 and x = 35
.
130
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
image of an open-ended can may be produced by synthesizing an image of a cylinder which
is constrained by two planes perpendicular to the cylinder. The constraints would specify
those points on the cylindrical surface which lie above the lower plane and below the upper
plane. As a matter of terminology, I will use the term quadric section to refer to a quadric
surface along with a set of constraints on that surface. In order to use 3dsyn, you must
understand the coordinate systems it uses to orient surfaces and sections. Theoretically,
you could place surfaces wherever you wanted by specifying the appropriate coefficients in
the quadric equation 7. In practice, this is a pain since the coefficients are a function of
the objects position and orientation as well as its shape. E.g.,
x2 + y 2 + z 2 = r 2
(4.8)
describes a sphere of radius r centered at the origin. The quadric coefficients are A = B =
C = 1, K = −(r2 ). Moving this sphere so that its center is at location (x, y, z) = (10, 5, 0)
gives: (x − 102 + (y − 5)2 + z 2 = r2 which when put into the form of equation 7 looks
like x2 + y 2 + z 2 ?20x?10y + (125?r)2 = 0. (10) Rotating an object affects the quadric
coefficients in a still more complicated way. 3dsyn allows you to specify a surface using
any coordinate system you desire; you may then translate or rotate the object to move it
to a different coordinate system. 3dsyn uses several different coordinate systems to ease
the task of creating images which are composed of multiple surfaces. Figure 3.1 shows the
various coordinate systems used, and are described here in the text.
Each surface (including constraints) is defined in terms of its own local coordinate
system. Typically you would choose the coordinate system in which it was easiest to
describe the shape you want. Each quadric section has its own coordinate system, known
as a section or object coordinate system. The latter term is perhaps misleading in that
what you think of as an object may actually consist of more than one section.
By specifying the relationship between the local coordinate system for each surface in a
section to the sections coordinate system, you specify how the various parts of the section
fit together. This essentially is used to relate the constraints to the actual surface being
synthesized, with the coordinate system of the surface itself (its local system) typically
being coincident with the section coordinate system. This is not a requirement though;
the surface and its constraints are all placed relative to a common section system, rather
than the constraints being placed relative to the local system of the surface.
The next higher level coordinate system is the reference or base coordinate system. This
is the base coordinate system for the entire image to be synthesized. Its relationship to the
object coordinate systems is the same as that of the object to local coordinate systems.
By specifying the relative locations of each object coordinate system in the base system,
you define how the various quadric sections fit together, and define what the overall image
will look like.
The highest level coordinate system is the view point coordinate system. This corresponds to the coordinate system for the image array, and hence, the display equipment.
4.2. 3DSYN-SYNTHESIZE DENSITY IMAGES
131
The relation between the view coordinate system and the base coordinate system specifies the position which you are actually seeing the image from. In many cases the two
coordinate systems may be coincident, or one merely a translation of the other. Note that
the term viewpoint system is somewhat misleading in that this is not a range image; this
transform really defines a plane of projection for the image (which is fully described at
the level of the base system). The projection plane itself is the xy plane of the viewpoint
coordinate system. The image array itself is just a finite piece of this infinite projection
plane. Specifically, the origin [x, y, z] = [0, 0, 0] corresponds with the voxel at location [f
rame, col, row] = [0, 0, 0] in the image array (which is in one corner of the image). Hence,
if you define the objects in your image to lie around the origin (in the base coordinate
system), when you display the image you will probably find that all of the objects will
lie in one corner of the image, unless you have displaced the base system relative to the
viewpoint system. Put simply, the origin of the base system will be in one corner of the
image you synthesize unless you make sure to move it - and you may end up not seeing
parts of your objects since they will be clipped at the image borders.
In 3dsyn, the relation between coordinate systems is expressed in terms of six basic motions: translations along the three coordinate axes, and rotations about the axes. Motions
along the axes go by the names of movex, movey, and movez. Rotations about the axes go
by the names of rotx, roty, and rotz, or alternatively as yaw, pitch, and roll, respectively.
When specifying the relationship between two coordinate systems, the motions are given
in terms of the higher coordinate system, i.e., the higher level system is the base system.
For example, to specify that the local coordinate system for a constraint has its origin at
location (x,y,z) = (10,5,2) in the section coordinate system, you would specify the motion
as (movex 10, movey 5, movez 2). Note that you do not specify the motion as (movex -10,
movey -5, movez -2), as this would be specifying the origin of the higher system (the section
system) in terms of the lower (the local system). This is easy to see for motions which are
pure translations, but may provide a source of confusion when rotations are involved.
The best way to regard the motions is as being object oriented. Although you are
specifying the relationship between two coordinate systems, the lower coordinate system
can be regarded as an object in the higher coordinate system (imagine that the lower level
system has a cube sitting at its origin, and you are moving the cube around in the higher
level system). Initially, the two coordinate systems start out with coincident axes, i.e., the
lower system is an object sitting at location (0, 0, 0) in the higher system. If you specify
a motion of movex 10, then all objects in the higher level system are displaced 10 units
down the x axis. If you specify a motion of roll 20, then all objects swivel around the z
axis of the higher system. This is not the same as simply rotating the lower coordinate
systems coordinate system by 20 degrees! If the lower systems origin coincides with the
higher systems origin, then the effects are the same; however, if the origin points do not
coincide, then the lower systems origin will be seen to swing on an arc around the higher
systems axis. Hence, the rotation will also cause a translation in two of the axes of the
higher system. Figure 3.1 illustrates this, and also shows how the order of motions is
132
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
important. 3dsyn works by reading a file which describes the image to be synthesized.
This file contains the quadric coefficients for each surface and constraint (in their own
local coordinate system), the density of the material contained within that surface, and
movement commands which specify how the coordinate systems are located. 3dsyn reads
this specification file and generates the image; when it is done, it prompts for the name of a
file to write the image to. The output file is a 3-d IFS format image. The data type for the
image will be unsigned byte, with the value of a voxel indicating its density. The format
for the image specification file is relatively simple. Figure 3.8 shows a sample specification
file; the format (described below), is identical to the description for Qsyn except for the
inclusion of density value on each line describing a quadric.
It is a text file composed of several blocks. These blocks may contain other blocks
within them. At the beginning of the file is a header block which specifies the size (number
of frames, rows and columns) of the image to be generated, the number of quadric sections
to synthesize, and a set of motions which will translate the reference coordinate system
to the view point coordinate system. The motions which specify the relationship between
the two coordinate systems are actually an instance of a type of sub-block known as a
coordinate transform block, or just transform. These transforms occur in several places,
and always have the same format. There are six basic motions which may be specified in
a transform block. There are also several complex motions which are simply composites
of the basic motions. All of the motions are specified by some keyword describing the
motion to perform, followed by the parameters appropriate for that keyword. The six
basic motions are those mentioned earlier: movex, movey, movez, rotz (roll), roty (pitch),
and rotx (yaw). The movex, movey, and movez commands can also be shortened to x, y,
and z. Each motion takes a single argument which is the amount to move or the angle to
rotate (in degrees). To specify a complete transform block, you merely specify an arbitrary
number of basic motions (in arbitrary order), followed by the keyword end. The basic
motions are performed in the order specified. A sample transform block might look like:
movex 20 movey 10 pitch 30 end
This will shift an object 20 units along the x axis, 10 units in y, and swing the object
30 degrees around the y axis.
The composite motions are just shortcuts for specifying certain common sets of motions. The com- bination movex10movey20movez30 can be specified more rapidly as
movexyz102030 or just xyz102030. Similarly, rpy102030 is short for roll10pitch20yaw30. All
six basic motions can be expressed using the commands rpyt ?z?y?xDeltaxDeltayDeltaz
and trpy ?z?y?xxyz. Rpyt does the rotations first, then the translations; trpy performs
the translations first. Note that the syntax for trpy is inconsistent in that although the
translations are performed first, they are the last arguments specified for the command.
After the header block there comes a set of section blocks, one block for each quadric
section to synthesize. Each block describes a surface to generate, and all the constraints
for the surface. The quadric section blocks are in turn composed of smaller blocks. The
first sub-block is a transform block which converts from the section coordinate system to
4.3. MATTE - SYNTHESIZE LUMINANCE IMAGES
133
32 32 32 7 rpyt 0.0 0.0 0.0 15.0 15.0 15.0 end
end 0.7164 0.4030 0.4030 0 0 0 0 0 0 -87.3151 255.0 0.0 end 0
rpyt 0.0 0.0 0.0 0.0 -0.2944 -0.2944 end 0.5835 0.3352 0.3352 0 0 0 0 0 0 -65.5430 -255.0 0.0
end 0
rpyt -18.0 0.0 0.0 3.52 0.0 0.0 end 0.00923521 0.00116281 0.00116281 0 0 0 0 0 0 -0.028607 31.0
end 0
rpyt 18.0 0.0 0.0 -3.52 0.0 0.0 end 0.02825761 0.00430336 0.00430336 0 0 0 0 0 0 -0.1851891 31.
rpyt 0.0 0.0 0.0 0.0 5.6 0.0 end 0.00390625 0.00275625 0.00275625 0 0 0 0 0 0 -0.04410 23.0 0.0
end 0
rpyt 0.0 0.0 0.0 0.0 1.6 0.0 end 1.0 1.0 1.0 0 0 0 0 0 0 -0.541696 47.0 0.0 end 0
rpyt 0.0 0.0 0.0 0.0 -1.6 0.0 end 1.0 1.0 1.0 0 0 0 0 0 0 -0.541696 47.0 0.0
end 0
Figure 13: Figure3.8: Example 3Dsyn input file (A synthetic head)
the base coordinate system. This is followed by a surface block. A surface block contains
a set of quadric coefficients to describe a single quadric surface, the interior and exterior
densities of the object, and also contains a transform block which translates the surfaces
local coordinate system to the section coordinate system. After the surface block comes
the number of constraints, followed by a set of constraint blocks. A constraint block is
identical to a surface block except that it contains a flag indicating which way the constraint
inequality goes (e.g., choose all points in the surface below some plane).
4.3
Matte - synthesize luminance images
CONVERSION TO LATEX NOT FINISHED Name: matte.c
Action: Produces a matte luminance image given a range image and one or more light
sources of any brightnesses. That is, the range image is modeled as being a Lambertian
reflector. matte converts a 2-D range image into a matte luminance image.
USAGE: usage: matte -i infile -o outfile -s
FLAGS: command line args are... -h help-screen -i Input file name -o output file name
-s Ignore shadowing LIGHTS: This program generates a matte image illuminated by a set
of light sources that you specify. The user is prompted for the locations of the light sources.
Notes
• Comments (like in the example) aren’t allowed
• Be careful with the z coordinate. Large positive z values place the light source in
front of the object: good. Negative z, or even small positive z, may place the light
BEHIND the object: (bad), which may generate a null image.
• 50 lights max.
134
CHAPTER 4. IMAGE SYNTHESIS PROGRAMS
Example:
matte -i pcblead.ifs -o leadbrightness.ifs
Enter number of light sources>2
For each source, enter position (col,row,height) and brightness>64 64 200 10 20 30 200 2
ALGORITHM
Read lights from user (or redirect of standard input) call ifsderiv to compute gradients
of range image for every pixel Compute normal to this pix; Compute vector from this
pix to each light; Compute cosine of angle between normal and lights; Light this pix by
summing the source brightness times the cosine.
4.4
Tomosim - simulate tomographic X-ray source
CONVERSION TO LATEX NOT FINISHED
This program is used to simulate a 3-D beam tomographic sensor. Either cone-beam or
parallel- beam sensors may be simulated. An ifs 3D image is used as input (as produced,
for example, by 3dsyn). The program produces a set of ifs 2D images, where each of the
output images corresponds to one projection USAGE:
tomosim inputimage numofpoints radius detector rows detector cols¡-o¿ ¡-d n¿ IFSIMG
input image; int numofpoints; /*center int radius; /*three dimensional*/ /*The number
of points around a 360 degree circle,*/ /*centered at the*/ of the volume specified by
the input image*/ /*The distance (in voxels) from the center of the*/ /*volume to the
detector. Will be treated as*/ /*identical as the distance from the */ /*center of the
volume to the center of the detector array*/ int detector rows; /*number of rows on the
detector*/ int detector cols; /*number of columns on the detector*/ The names of the
output files are read from stdin as they are needed. Since the program may take quite a
while to run, manually typing these names is tedious. The recommended way to run the
program is to first create a file containing the names of all the output image files, and then
to run tomosim redirecting stdin to this file. Switches -o The -o switch, if used, will use
parallel-beam (orthographic) rather than cone beam projection -d The -d switch, if used,
means that the next argument will be the debug control value (really only of interest to the
programmer) Application: For cone beam, just specify the number of points. The sensor
rotates in x-y plane, about an origin at the center of the 3D volume provided. For fan
beam, simply specify a detector with only one row. The -o switch provides parallel beam
simulation in either single row or multiple row cases.
Chapter 5
Programs for processing images
The following is a list of programs which exist in the ifs bin directory. (Depending of
local installations, this is usually /usr/local/bin/ifs) These programs are for the most part,
simply mains wrapped around some of the standard subroutines documented in earlier
chapters. These programs are only documented briefly here, since the operation is generally
obvious.
Generally, on-line help for any program can be obtained by simply starting that program
up, but providing it an incorrect number of arguments. Most programs also support the -h
switch. For example, add -h will tell you how to use add. Most programs us the -x switch
convention. For example,
program -i infilename -o outfilename
5.1
add
add two ifs images, point by point Author: Wes Snyder
5.2
addhdr
adds an IFS header to a raw data file. rmvhdr is the reverse function. Perpetrator: Mark
Lanzo
5.3
atoi
Converts an ascii file to ifs. Input file is to be in the format produced by itoa using the -v
switch. The -v switch on itoa adds two lines at the beginning of the file which specifies the
135
136
CHAPTER 5. PROGRAMS FOR PROCESSING IMAGES
size and data type. Author: Wesley Snyder
example:
Image has 4 rows and 4 columns
Data format is u8bit
0 0 0 0
0 1 0 0
0 0 2 0
0 0 0 0
/end{verbatim}
Usage:
\begin{verbatim}
atoi imga.txt imga.ifs
Note this program does not use the -i -o convention.
5.4
any2any
convert data types
Converts any IFS image to or from (almost) any other data format including other ifs data
types
usage: any2any [switches]
-i inputfile.ifs
-o outputfile.xxx
-d datatype
switches are
datatype is an ifs data type, such as byte, ubyte, char, uchar, u8bit, short, ushort, u16bit,
int, float 32flt, double, 64flt, etc. For other file types, such as bmp, jpg, png, etc, the type
is determined by the file name. If outfile is not ifs, dtype IS required to be uchar, (for this
version), even if neither in nor out is ifs.
Examples:
convert an ifs image of some unspecified data type to an ifs image of type float
any2any -i startrack.ifs -o startrackfloat.ifs -d float
convert an ifs image of unspecified data type to png. Note that this may not work if the
input if 3-D.
any2any -i startrack.ifs -o startrack.png -d uchar
convert a jpeg input image to a float ifs image.
any2any -i lena.jpg -o lenafloat.ifs -d float
5.5. C2IMAG
5.5
137
c2imag
take imaginary part of an ifs image, point by point Author: Wesley Snyder
5.6
c2mag
take magnitude of an ifs image, point by point Author: Wesley Snyder
5.7
c2phase
take phase of an ifs image, point by point Author: Wesley Snyder
5.8
c2real
take real part of an ifs image, point by point Author: Wesley Snyder
5.9
compmag
produces an ifs file (type float) equal to the log of the square of the magnitude of a complex
image Victim: Wesley Snyder
5.10
ChooseFrame
Select a single frame from an image. This is particularly useful for images which have been
read as color (say, JPEG) and you wish to store a single frame of the RGB image.
ChooseFrame -i inputimage -o outputimage -f framenumber
The default frame number is zero.
5.11
gauss add Gaussian-distributed noise to an image
Switches:
-i inputfilename
-o outputfilename
-m mean
-s stddev
-n (no argument) - don’t allow negative values for noise
-w (no argument) -don’t allow overflows (wrap arounds) e.g. adding a value of 3 to an 8bit
138
CHAPTER 5. PROGRAMS FOR PROCESSING IMAGES
pixel with value of 254 will produce a pixel with value of 1
-S IntegerSeedForRandomNumGen (normally dont use this)
5.12
ifsDoG Apply a derivative of Gaussian kernel to an ifs
image
usage: ifsDoG [switches]
switchs are
-i infilenname, where infilename is an ifs image default: the null string
-o outfilenname, where outfilename is an ifs image default: the null string
-s sigma, std. dev of the filtering Gaussian default: sqrt of 3.0
-O order of the derivative zero is blur with the Gaussian, one is first deriv, etc, up to 3rd
deriv direction.
-d This flag is the letter c for col, r for row, b for both default: columns
b option only valid for second deriv
example: ifsDoG -i myfile.ifs -o outfile.ifs -O 0 -d r
5.13
itoa
converts an image to printable ASCII. The output is written to the terminal. Output can
be redirected to a file using the Unix redirect command.
Examples:
1) to output an image to the screen:
itoa site14.c.ifs
2) to ouptut an image to a file
itoa site14.c.ifs > site14.c.txt
3) to output an image to a file with a prefix specifying the rows, columns, and data type
itoa -v site14.c.ifs
If the -v switch is used the first two lines output will look like this:
Image has 48 rows and 32 columns
Data format is 16bit
Author: Mark Lanzo
5.14. MKDOC
5.14
139
mkdoc
makes a LaTeX compatible version of this index on standard out. Author: Wesley Snyder
5.15
mult
multiply two ifs images, point by point Author: Wesley Snyder
5.16
profile
Take a cross section of an IFS 2D image. Output is in standard plot filter format (to
stdout). Author: Mark Lanzo
5.17
prthdr
Print the header structure for an IFS image (in human readable format). Author: Mark
Lanzo
5.18
rmvhdr
Remove the header from an IFS image to yield a raw data file. Author: Mark Lanzo
5.19
subsample
subsamples an arbitrary ifs image to be of a speci ed size: Author: Wesley Snyder
5.20
recip
take reciprocal of an ifs image, point by point Author: Wesley Snyder
5.21
sub
subtract two ifs images, point by point Author: Wesley Snyder
140
5.22
CHAPTER 5. PROGRAMS FOR PROCESSING IMAGES
triangulate
Find an optimal set of triangles which completely covers a set of points.
Usage:
triangulate inputimage outputimage
(Note that this program, at this time, does not use the standard -i, -o designation for
arguments. Therefore, the order of the arguments IS important.) This program accepts an
input file, a standard text file, which contains the set of row, column coordinates of vertex
points. The output file will contain descriptions of triangles which cover that set of points.
5.22.1
Input File Format
First line, the number of vertices
all other lines, row, column pair
example:
4
0
1
1
0
0
0
10
10
5.22.2
Output File Format
First line: number of triangles found.
all other lines, row, col coordinates of the vertices in this triangle
example:
2
0 10 0 0 1 0
1 0 1 10 0 10
5.23
vidscale
video scale an ifs image, that is, produce an output image which is the same size as the
input image, but scaled to be between 0 and 255, and be unsigned char.
vidscale -i infile.ifs -o outfileu8bit.ifs -s
5.24. WINDOW
5.24
141
window
This program extracts a window from an ifs image. The resultant output image is of the
same data type as the input.
Usage:
window input output xleft ylower xright yupper
input and output are two dimensional ifs image files. Output will be created by this
program. xleft is the index of the left-most column of the input image which should be in
the window. ylower is the index of the lowest-index row of the desired window. xright and
yupper are the other extremes. NOTE: yupper must be greater than ylower. Thus, upper
and lower correspond to indices, not to a top-bottom relation on a display screen. Author:
Wesley Snyder
5.25
zoomupdown
This is a program which is really just a wrapper called the iptools function by the same
name.
Usage: zoomupdown inputimage outputimage numrowsofOutputImage numcolsofOutputImage
Program will read the input image and will create an output image with the specified
number of rows and columns. Then the input image will be resized to be of the specified
number of dimensions. No interpolation is done across frames, in the case of 3D images.
If you need to do that, use squp. If the input image is 3D, and has 3 frames, it will be
treated as if it were color.
142
CHAPTER 5. PROGRAMS FOR PROCESSING IMAGES
Chapter 6
Programs for Analyzing Images
In this section, programs are provided which do image analysis.
6.1
Harris-Laplace
This program reads an image file and computes the output of the Harris corner detector
over scale, and applies a Laplacian over the same range of scales.
Usage:
HarrisLaplace [switches]
Switches are:
-i inputfilename
-H Harris measure out. Default: HarrisMeasure.ifs
-L Laplace measure out. Default: LaplaceMeasure.ifs
The output images are ifs 3D images, data type of these images is float.
6.2
PCON - Piecewise-constant image noise removal
This program does Edge-preserving smoothing and noise removal using a piecewise-constant
prior.
USAGE:
pcon [switches] .
Switchs are -g input filenname, where infilename is an ifs image; default: the null string
-s sigma The std dev of the noise. default: 1.0
-r r step size, smaller is more stable but slower. default: 0.02
143
144
CHAPTER 6. PROGRAMS FOR ANALYZING IMAGES
-b b The coefficient in front of the prior term. Beta of Eq. 6.41 in Snyder and Qi
-t t1 initial temperature. default: 2.0
-T t2 final temperature default: 0.0005
-l dlt rate by which the temperature drops. default: 0.99
-o outcount output a temporary file every this many iterations. default: 50
example: pcon -g foofile.ifs -s 3
SEE Snyder and Qi, Machine Vision, Cambridge University Press, 2005, Equation 6.41.
This program works by finding the image f which minimizes
X (fij − gij )2
−|∇|ij
bX
−
(6.1)
exp
2σ 2
τ
2τ 2
ij
ij
where g is a measured image, σ is the standard deviation of the noise, and the notation
|∇fij | denotes some scalar measure of the image gradient in the vicinity of pixel ij. There
are five parameters to this program:
• sigma which is the standard deviation of the (assumed additive, Gaussian, white)
noise.
• b, the strength of the second term, relative to the first. The second term is called the
“prior” term, since it is a function of only the (unknown) true imagef and not of the
measured, noisy image g.
• τinitial is the initial value of τ . In most literature, the variable τ is called the temperature because the algorithm bears similarity to the process of annealing in metallurgy.
The algorithm implemented in pcon is an approximation to simulated annealing,
called “mean-field annealing.” The optimal value of τiniital is problem dependent,
but a good approximation can be found by performing the following analysis: Find
the largest pixel-to-adjacent-pixel difference which should be considered noise and
not an edge. Set τiniital to twice that value.
• τf inal is the final of τ at which the algorithm will stop. I suggest you allow the
temperature difference be two-to-four orders of magnitude. E.g., if τinitial is 10.0,
τf inal should be something like 0.01.
• λ is the rate of decrease of temperature. The program uses a geometric annealing
schedule (as opposed to a logarithmic schedule) following τ = λτ . A good value is
typically 0.99. Values closer to one (e.g. 0.999) will run slower but give better results.
Example:6.1 illustrates a noisy image, a cardiac angiogram, to which pcon is applied to
produce figure 6.2. Notice that some bright points are preserved. These are points where
the random noise was sufficient to make the pixel-to-pixel difference significantly larger
than τinitial , therefore these points are considered by the algorithm to be edges and are
6.2. PCON - PIECEWISE-CONSTANT IMAGE NOISE REMOVAL
Figure 6.1: Input image
Figure 6.2: Piecewise-constant image
145
146
CHAPTER 6. PROGRAMS FOR ANALYZING IMAGES
Figure 6.3: Profile across an artery, showing the piecewise-constant property of the solution
not removed. The profile illustrated in Figure 6.2 across one of the arteries in the output
shows the piecewise- constant result.
The implementation of PCON is optimized for speed, using pointer arithmetic and
other programming hacks.
6.3
PLIN - Piecewise-linear image noise removal
This program does Edge-preserving smoothing and noise removal using a piecewise-linear
prior.
USAGE:
plin -i infile [-s sigma] [-r stepsize] [-b priorgain] [-t initialtemp]
[-T finaltemp] [-l dlt] [-o n]
Once the program is running, it will prompt the user for control variables, these are listed
below, along with good values to use
sigma: The std dev of the noise. default 1.0
r,step size, smaller is more stable but slower. default is 0.02
b, The coefficient in front of the prior term. Beta of Eq. 6.41 in Snyder and Qi,
good value is .5
t,
initial temperature. default
2.0
6.3. PLIN - PIECEWISE-LINEAR IMAGE NOISE REMOVAL
147
t2, final temperature default 0.005
lambda: rate by which the temperature drops. good value : 0.99
n: used with the -o switch. The program will output a temporary
image every n iterations
example: plin -i G
SEE Snyder and Qi, Machine Vision, Cambridge University Press, 2005, Equation 6.41.
This program works by finding the image f which minimizes
X (fij − gij )2
−|∇2 |ij
bX
−
exp
2σ 2
τ
2τ 2
ij
(6.2)
ij
where g is a measured image, σ is the standard deviation of the noise, and the notation
|∇fij | denotes some scalar measure of the image gradient in the vicinity of pixel ij. There
are six parameters to this program:
• sigma which is the standard deviation of the (assumed additive, Gaussian, white)
noise.
• r, The step size of the gradient descent term.
• b, the strength of the second term, relative to the first. The second term is called the
“prior” term, since it is a function of only the (unknown) true imagef and not of the
measured, noisy image g.
• τinitial is the initial value of τ . In most literature, the variable τ is called the temperature because the algorithm bears similarity to the process of annealing in metallurgy.
The algorithm implemented in pcon is an approximation to simulated annealing,
called “mean-field annealing.” The optimal value of τiniital is problem dependent,
but a good approximation can be found by performing the following analysis: Find
the largest pixel-to-adjacent-pixel difference which should be considered noise and
not an edge. Set τiniital to twice that value.
• τf inal is the final of τ at which the algorithm will stop. I suggest you allow the
temperature difference be two-to-four orders of magnitude. E.g., if τinitial is 10.0,
τf inal should be something like 0.01.
• λ is the rate of decrease of temperature. The program uses a geometric annealing
schedule (as opposed to a logarithmic schedule) following τ = λτ . A good value is
typically 0.99. Values closer to one (e.g. 0.999) will run slower but give better results.
148
CHAPTER 6. PROGRAMS FOR ANALYZING IMAGES
Figure 6.4: Piecewise-linear image
Example:6.1 illustrates a noisy image, a cardiac angiogram, to which plin is applied to
produce figure 6.4. Notice that some bright points are preserved. These are points where
the random noise was sufficient to make the pixel-to-pixel difference significantly larger
than τinitial , therefore these points are considered by the algorithm to be edges and are
not removed. The profile illustrated in Figure 6.5 across one of the arteries in the output
shows the piecewise- linear result.
The implementation of PLIN is optimized for speed, using pointer arithmetic and other
programming hacks.
6.4
WaterSheds
The program WaterShed is a wrapper around the function ifs WaterShed, allowing the
function to be called from the command line. In addition, the user may select several
post-processing features.
Usage:
WaterShed -i infilename -o outfilename -c connectedtype [-C filename] [-n filename]
outfile will contain watershed points
-c is connection type, either 4-connected or 8-connected.
-C file will be in color
-n file will not have watershed points
6.5. CCL CONNECTED-COMPONENT LABELING
149
Figure 6.5: Profile across an artery, showing the piecewise-linear property of the solution
The input image must be unsigned char ifs image.
The output image will be int.
The program first finds all minima in the image and assigns each a label. Those labels
become the labels of regtions.
Figure 6.6 illustrates an image, with a surround which is minimal, a brightness which
grows positive and then drops in two places (a volcano with a vent). Figure ?? shows
the watershed output, illustrating the presence of watershed pixels. Figure 6.7 illustrates
a labeling with all the watershed pixels assigned to regions, and Figure ?? shows a color
labeling of that image.
6.5
ccl Connected-component Labeling
This program uniquely labels the connected components of an ifs image. It uses the algorithm described in Snyder and Qi, Machine Vision, Cambridge University Press, 2005.
This program only supports images in the following data types: unsigned char, short,
and float. Switches:
-h : print out this message
-i <input_image_file_name>
-o <output_image_file_name> This will be the label image, in which is stored at each pixel
-p <parameter_file_name> (optional)
-l <lower limit for labeling region> (default 0) Normally this is the background
-u <upper limit for labeling region> (default 255)
-[f|e|v] : use face/edge/vertex connectivity (default face)
150
CHAPTER 6. PROGRAMS FOR ANALYZING IMAGES
Figure 6.6: Left:Original image (Volcano with a vent and a zero surround) Right: watershed
segmentation showing watershed pixels
Figure 6.7: Left:Region Labeling with watershed pixels assigned to regions. Right: Region
Labeling in pseudocolor, using a random color map
6.6. RESAMPLE CURVE
151
-t <threshold for connectivity>
(two pixels which differ by more than this will not be merged-(default 2)
-s <row> <column> <frame> (Seed Location of Region of Interest)
Normally, ROIs are not used. -r <label of ROI> (default 1)
-g : outimg is ROI_seg_out image (default label image)
-j <pixel value for ROI_seg\_out image > (default 128)
-a <upper background label> (default 0)
-b <lower background label> (default 0)
-m <minimum label used> (default 0)
-x <maximum label used> (default 32767)
-c <Size of the content-addressable memory used to resolve
equivalent region labels.> (default 4096)
ccl is a wrapper around the function ifs ccl. It parses the command line, fills in the structure
used by ifs ccl, initializes some things, and then calls ifs ccl. If you need to call connected
components from inside a program, see the description of the function ifs ccl. See the
documentation for ifs ccl for additional documentation, including figure 3.1 which defines
the types of connectivity.
6.6
resample curve
The program resample is simply a wrapper for resampleinit and resamplesub. See section
3.1.6. It resamples a discrete curve and returns a set of points equally far apart (arc-length
sampling).
usage:
resample -i infile -o outimage -n numpoints
inputfile is an ascii file of x-y pairs.
the first line is a sharp sign, a space, and the number of pairs, e.g. # 798
The output file will be written in the same format, but the numbers will be float
152
CHAPTER 6. PROGRAMS FOR ANALYZING IMAGES
Chapter 7
Programs for Viewing IFS images
7.1
ColorMap
The program ColorMap allows you to assign a number of different colors to an 8-bit,
unsigned image. That is, each of the possible 256 brightness levels will be assigned a color.
There are many ways to assign an R, G, and B value to a single pixel brightness. The
program offers several different ways to interpret the brightnesses.
Usage:
ColorMap -i inputfile.ifs -o outputfile.ifs -m X -f Y
Where inputfile is an ifs image. and X is an integer between 0 and 6 which tells the program
which color map to use:
0 grey scale
1 inverse gray scale
2 bronson
3 hot metal
4 heated spectrum
5 log
6 random
The input image may be any ifs format, however, if it is not unsigned 8 bit, the user will
get a warning message that only brightnesses between 0 and 255 will be mapped. If the user
has this problem, he or she should probably use vidscale to convert the darkest-brightest
values to 0-255.
Should the input file be 3D, the program will convert frame 0 by default. Using the -f
switch, the frame to be converted can be selected.
The output file will the 3 D ifs image, with 3 frames, with the usual interpretation that
frame 0 is red, frame 1 is green, and frame 2 is blue.
153
154
7.2
CHAPTER 7. PROGRAMS FOR VIEWING IFS IMAGES
Eigenimages
Program to find the principal eigenimage of a set of images.
The program will optionally also calculate the projection of another image onto the principal eigenimage.
Usage:
Eigenimage -c controlfile -o outfile [-t projfile]
The control file contains a single number (number of files followed by a list of file names,
one per line.
Example:
2
file1.ifs
file2.ifs
file3.ifs
The output file is an ifs image of the principal eigenimage.
The -t switch allows the user to specify the name of a file to project onto the eigenimage,
and measure the length of the projection
All images must have the same dimensions.
7.3
ifsview – view ifs images
This is a new program with ifs release 8. It supercedes wxifsview, which may still run on
some computers.
The application ifsview provides the ability to view, zoom, or probe ifs images. It uses
the openCV software system. To run ifsview, use the following command:
ifsview -i image [-c] [-I ignoreswitch] [-s]
image is the input image. image can be any format of ifs, or jpg, tiff, bmp, and png. If ifs,
it can be monochromatic or color. If monochromatic, it may have two or three dimensions.
colorswitch The optional color switch, if it exists, tells ifsview that if the input image
happens to be three dimensional and have three frames, it should be displayed as a color
image, rather than a 3D image.
ignoreswitch If this brightness is entered, then any pixels of this value will not enter the
calculation of contrast.
-s If the -s switch is present, frames in a 3D image with be brightness-scaled independently.
Otherwise, the brightest and darkest points in the entire image will be used for scaling.
Once you have started up ifsview, you have a variety of options:
7.3. IFSVIEW – VIEW IFS IMAGES
155
• LOUPE As soon as the image pops up, and you move the mouse over it, another
image will pop up which is a zoomed version of the original image, zoomed about
where the mouse is. We will refer to this second image as the “loupe” image, as in a
jewler’s loupe. As you move the mouse, the second image will move around also.
• stopping the tracking A right click on the original image will freeze the loupe image. A
second right click will turn tracking back on. On the mac, a right click is accomplished
by hitting the trackpad with two fingers simultaneously.
• Additional zoom With the pointer on the original image, you can use right and left
arrow keys to increase or decrease zooming (respectively). Note that you will not see
the result of the zoom until you move the mouse again, which will refresh the loupe
image.
• Image values – original image Clicking on the original image will return the brightness
at the point of the click, or if color, will return the R,G,B values.
• Image values – loupe image To look at points in great detail, first move the mouse
to get to where you want to be in the original image. Then right click to turn off
tracking. Move the pointer to the loupe image, and click on any point. The brightness
or color of that point will be returned.
• Exploring 3D images With the mouse on the original image (tracking can be on or
off), the up and down arrow will display different frames of the 3D image. Remember
that these values don’t show up in the loupe image until you move the mouse. On
some computers, the arrow keys don’t work with this application. As an alternative
to the arrow keys, you can use
m
i
j
k
c
control-c
L
l
frame up
frame down
zoom down
zoom up
toggle contrast enhancement on/off
exit program
increase size of loupe image
decrease size of loupe image
• Contrast enhancement The lower-case c key will toggle contrast enhancement of the
loupe image on and off.
Should you simply wish to display an image and obtain information about it, use the
program ifsview, which is just a wrapper around ifsCVPopup, and that way, you won’t
need to mess with linking openCV, which can be a painful hassle.
IMPORTANT NOTE ABOUT ifsview:
156
CHAPTER 7. PROGRAMS FOR VIEWING IFS IMAGES
ifsview uses the openCV functions. OpenCV supports ONLY dynamic libraries. This
means you must either have openCV installed on your computer, or you must have the
openCV libraries available, even though you actually aren’t writing any software that will
use them. We tried (unsuccessfully) to get openCV to generate static libraries so it would
be portable, but we failed, and learned from web conversations that it does not seem to be
possible. The openCV libraries are in the ifslib that you downloaded.
7.4
makecolor
This program accepts a 2D ifsimage as input and produces a jpg, png, or other output.
This is necessary because ifs uses openCV to convert images from ifs to other format.
openCV thinks all jpg, png, etc. images are color, so if you give it an input which is a 2D
image, it will fail, producing an output which is not readable.
Usage: example:
makecolor -i input.ifs -o output.ifs
7.5
PlayMovie- play a 3D ifs image as a movie
Plays a 3d ifs image as a movie
Usage:PlayMovie [arguments]
switches are
-i inputfile.ifs
-z floating point zoom factor. Will zoom image prior to playing. Default: 1.0
-h this usage
-s time (in microseconds) to delay between frames (minimum 1, max 999999). If the
argument of the -s switch is greater than 1,000,000, the program will wait for the user to
hit a key before going to the next frame.
-d direction if direction is 0, the movie will be played forward, if 1, backward, if 2 forward
then back.
This program is just a wrapper around CreateIFSDisplayWindow and WriteToIFSDisplayWindow. It plays a 3D image as a movie with controllable zoom and speed. IMPORTANT NOTE ABOUT PlayMovie:
PlayMovie uses the openCV functions. OpenCV supports ONLY dynamic libraries.
This means you must either have openCV installed on your computer, or you must have
the openCV libraries available, even though you actually aren’t writing any software that
will use them. We tried (unsuccessfully) to get openCV to generate static libraries so it
would be portable, but we failed, and learned from web conversations that it does not seem
to be possible. The openCV libraries are in the ifslib that you downloaded.
7.6. PLOTCONTOUR
7.6
157
Plotcontour
Plotcontour: a program which will plot a 1D contour into an ifs image. The contour will
be white on a black background. usage:
plotcontour -i infile -o outimage -s scale
-r imagerows -c imagecols
infile is an ascii file of x,y point pairs, the first line of which is sharp sign, space,
number of points
e.g. # 799
On each following line, the format is ”x y”
Normally, x is columns and y is rows
example:
# 4
0 200
200 0
0 0
200 200
the outimage file is the name of an ifs file containing white dots on a black background
optional argument scale will multiply x and y coords by scale
optional arguments imagerows, imagecols will force the image output to be that size.
Note that if the points go off the image, they will be clipped.
the -p switch will write precisely the coordinates specified in the file.
7.7
C2LaTeX-convert source code to LaTeX
This program allows the user to include his/her source code into a LaTeX document. Any
line which starts with
//l(the ell character)
will not only be a comment to C or C++, but allows the user to include any LaTeX command at all. (Well, almost any). The exception is you may not include the string
\end{verbatim}
You may of course, use the \textbackslash with no problem. You might also consider
using the listing environment rather than the verbatim environment.
Usage:
C2LaTeX -i inputfile.c -o outputfile.tex -m ValueOrFilename
158
CHAPTER 7. PROGRAMS FOR VIEWING IFS IMAGES
if the -m switch as a value of zero: (e.g. below)
C2LaTeX -i inputfile.c -o outputfile.tex -m 0
Then the output LaTeX file will not contain a header (the stuff before the \begin{document})
or the \end{document} . This is particularly useful if you intend to include or input this
LaTeX file into another file.
If it is to be stand-alone, the -m switch should specify the name of the header file you
want the program to use, e.g.:
C2LaTeX -i inputfile.c -o outputfile.tex -m myheader.tex
				
											        © Copyright 2025