-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathv_btabpds.pro
executable file
·522 lines (456 loc) · 20.1 KB
/
v_btabpds.pro
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
function V_BTABPDS, filename, label, Object, SILENT = silent, C_name = columns
;+ $Id: v_btabpds.pro,v 1.9 2013/11/05 13:20:06 sophie Exp $
;
; NAME:
; V_BTABPDS (binary-table PDS)
; PURPOSE:
; Read a PDS binary table object into an IDL structure.
; Reads a single table identified by pointer and label lines
;
; CALLING SEQUENCE:
; Result = V_BTABPDS (Filename, Label, Object [,/SILENT] )
;
; INPUTS:
; Filename = name of the file containing the PDS label
; (if detached label, this is the label file)
; Label = String array containing the PDS label itself (returned by v_headpds)
; object = Object definition limits in label + data pointer (returned by v_objpds)
;
; OUTPUTS:
; Result = Structure with fields:
; .columnN = N vectors containing the PDS table (can be of various types)
;
; OPTIONAL INPUT KEYWORDS:
; C_name: returns column names in a vector
; SILENT - suppresess console messages
;
; EXAMPLE:
; Read a single binary table, the i-th object in the label, in a variable:
;
; Obj_def = V_OBJPDS(label, /all)
; Obj_num = (size(obj_def, /dim))(0) ; number of objects found
; data = V_BTABPDS(filename, label, Obj_def(i), SILENT = silent)
;
; COMMENTS:
; Requires correct label formatting (ROW_BYTES is correct)
; Does not require the last data record to be completed.
; Does not require column names.
; Does not support external format definition (^Structure ).
; I/O format is different from the original SNBIDL routine tbinpds.pro
; Result is still a structure to handle tables containing different variable types
; Won't work with external column format (see below; fails with container objects...)
;
; modification history (v_tbinpds):
; Adapted by John D. Koch from READFITS by Wayne Landsman,December,1994
;
; 10/6/99 M.Barker - Fixed bug that miscalculated offset.
; 10/15/99 M.Barker - Added support of scaling and offset factors.
; Modified for VIRTIS, Stephane Erard, oct. 99
; Stephane Erard, sept. 2000: Updated from SBNIDL 2.0,
; Fixed search for object pointer
; Stephane Erard, LESIA, Dec 2005:
; Fixed scaling process
; Added search for INDEX_TABLE, std PDS object
; Changed access to data object (offset...)
; Changed type/endian conversion (more complete)
; Stephane Erard, LESIA, Jan 2006:
; Added support for DATA_TABLE + default names if column names not present
;
; MODIFICATION HISTORY (v_btabpds):
; Stephane Erard, LESIA, Feb 2006
; Derived from v_tbinpds with different I/O
; Now read any table in file, given pointer to the object
; Fixed structure length for I/O (solves rare EOF errors depending on dimensions)
; Stephane Erard, LESIA, June 2006
; Fixed column number...
; Now performs IEEE float swapping (required to read floats on Intel)
; Stephane Erard, LESIA, Jan 2007
; Fixed interchange format handling
; Stephane Erard, LESIA, Feb 2007
; Changed handling of detached labels. Fixed byte conversions.
; Stephane Erard, LESIA, Sept 2012
; Removed tentative reading of external file if STRUCTURE kw present in label
; (arrrived here, any label extension is already included in the label)
; Fixed label parsing in multitable mode
; Handles 6 bytes floats (in M3 files): retuns 6 bytes chains
; Restored conservative column count (in case of column # error in label, ie M3)
; Stephane Erard, LESIA, Oct 2012
; Fixed an old bug about little endian support (was apparently always assuming MSB)
; Now support table prefix/suffix, return them in the output structure
; Now returns if table contains bit_column objects (not supported)
;-
;
;###########################################################################
;
; LICENSE
;
; Copyright (c) 1999-2008, StŽphane Erard, CNRS - Observatoire de Paris
; All rights reserved.
; Non-profit redistribution and non-profit use in source and binary forms,
; with or without modification, are permitted provided that the following
; conditions are met:
;
; Redistributions of source code must retain the above copyright
; notice, this list of conditions, the following disclaimer, and
; all the modifications history.
; Redistributions in binary form must reproduce the above copyright
; notice, this list of conditions, the following disclaimer and all the
; modifications history in the documentation and/or other materials
; provided with the distribution.
; Neither the name of the CNRS and Observatoire de Paris nor the
; names of its contributors may be used to endorse or promote products
; derived from this software without specific prior written permission.
;
; THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND ANY
; EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
; WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
; DISCLAIMED. IN NO EVENT SHALL THE REGENTS AND CONTRIBUTORS BE LIABLE FOR ANY
; DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
; (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
; ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
; (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
; SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
;
On_error,0 ;2, Return to user
; ------- Check for filename input
if N_params() LT 2 then begin
print,'Syntax - result = BTABPDS( filename, lbl, Object [,/SILENT])'
return, -1
endif
silent = keyword_set( SILENT )
fname = filename
; object definition area in label
lbl_obj = label(object.start:object.stop)
lbl_obj = [lbl_obj, ' '] ; extra line for future use with v_parpds
; ------- To do: Check type of arguments
; ------- Determine type of data in file
; get exact type in objpoint
objpoint = Object.type
; return and continue if table contains bit_column objects
temp = v_pdspar( lbl_obj, 'OBJECT' )
bid = where(temp EQ 'BIT_COLUMN', cts)
if cts NE 0 then begin
if ~silent then message,'WARNING - '+fname+' BIT_COLUMN object not supported', /cont
columns = " "
return, -1
endif
bid = where(temp EQ 'ALIAS', cts)
if cts NE 0 then begin
if ~silent then message,'WARNING - '+fname+' ALIAS object not supported in tables', /cont
columns = " "
return, -1
endif
; check formatting
inform = v_pdspar( lbl_obj, 'INTERCHANGE_FORMAT' )
if !ERR EQ -1 then begin
message,'ERROR - '+fname+': missing required INTERCHANGE_FORMAT keyword', /cont
return, -1
endif else begin
inform = inform(0)
infst = strpos(inform,'"') ; remove '"'s from inform
if infst GT -1 then $
inform = strmid(inform,infst+1,strpos(inform,'"',infst+1)-infst-1)
if inform EQ "ASCII" then message, $
'ERROR- '+fname+' is an ASCII table file; try ATABPDS.'
endelse
record_bytes = v_pdspar(label,'RECORD_BYTES')
if !ERR EQ -1 then begin
message, 'ERROR - '+fname+' missing required RECORD_BYTES keyword', /cont
return, -1
endif
; M.Barker 10/8/99:
; Look for scaling and offset factors. These are not required, so if none
; found, then we must make sure future uses of !ERR are not confused.
scale_flag=0
offset_flag=0
scale = v_pdspar(lbl_obj,'SCALING_FACTOR',INDEX=sc_ind)
if !ERR EQ -1 then !ERR = 0 else scale_flag=1
offset = v_pdspar(lbl_obj,'OFFSET',INDEX=off_ind)
if !ERR EQ -1 then !ERR = 0 else offset_flag=1
;-------------------------------------
; @SE2012: Removed - already done in v_headpds
; get column format
; (should check that all columns are properly documented)
; (if not, should look for a ^structure pointer)
; Struct = v_pdspar(lbl_obj,'^STRUCTURE')
; Struct = strcompress(strupcase(Struct),/rem) ; make upper case, no space
; Struct = (strsplit(Struct, '"', /extract))(0)
; if not silent then print, 'Reading external file', struct
; if !ERR NE -1 then begin
; message, 'External table format not implemented', /cont
; !ERR = -1
; return, 0
; temp = v_headpds(Struct) ; just a try - won't work with CONTAINER
; lbl_obj = [lbl_obj, temp]
; endif
; columns = v_pdspar(lbl_obj,'COLUMNS')
columns = v_pdspar(lbl_obj,'COLUMNS',COUNT=cols)
if !ERR EQ -1 then begin
message,'ERROR - '+fname+': missing required COLUMNS keyword'
endif else columns = columns(0)
data_type = v_pdspar(lbl_obj,'DATA_TYPE',COUNT= dcount,INDEX=typ_ind)
if !ERR EQ -1 then begin
message, 'ERROR - '+fname+' missing required DATA_TYPE keywords', /cont
return, -1
endif
length = v_pdspar(lbl_obj,'BYTES',COUNT=bcount,INDEX=len_ind)
if !ERR EQ -1 then message, $
'ERROR - '+fname+' missing required BYTES keywords'
; col_start = v_pdspar(lbl_obj,'START_BYTE',COUNT=cols,INDEX=st_ind) - 1
col_start = v_pdspar(lbl_obj,'START_BYTE',INDEX=st_ind) - 1
if !ERR EQ -1 then message, $
'ERROR - '+fname+' missing required START_BYTE keywords'
name = v_pdspar(lbl_obj,'NAME',INDEX=nam_ind)
if !ERR EQ -1 then begin
if ~silent then message,'WARNING - '+fname+' missing required NAME keywords', /cont
name = 'column'+indgen(cols+1)
endif
;cols = cols(0)
;cols = columns ; number of columns provided by kw
cols = N_elements(col_start) ; number of columns actually described - use this to avoid inconsistencies!
if cols NE columns then $
if ~silent then message,'WARNING - '+fname+' column number mismatch', /cont
; Look for row prefix/suffix bytes in table object only (@SE2012)
prefix_bytes = v_pdspar( lbl_obj,'ROW_PREFIX_BYTES',COUNT=Precount,INDEX=Preind)
if precount EQ 0 then prefix_bytes = long(0)
suffix_bytes = v_pdspar( lbl_obj,'ROW_SUFFIX_BYTES',COUNT=Precount,INDEX=Preind)
if precount EQ 0 then suffix_bytes = long(0)
; Check to see if there may be an 'array-column' in the file
arrays = 0
items = v_pdspar(lbl_obj,'ITEMS',COUNT=arrays,INDEX=is_ind)
if !ERR GT -1 then begin
item_bytes = v_pdspar(lbl_obj,'ITEM_BYTES',COUNT=iarrays,INDEX=ib_ind)
if !ERR GT -1 then begin
if iarrays NE arrays then message,$
'ERROR - '+fname+': ITEMS count and ITEM_BYTES count discrepancy'
length = [temporary(length),item_bytes]
len_ind = [temporary(len_ind),ib_ind]
endif
if dcount(0) LT cols then begin
item_type = v_pdspar(lbl_obj,'ITEM_TYPE',COUNT=iarrays,INDEX=it_ind)
if !ERR EQ -1 then message,$
'ERROR - '+fname+' missing required ITEM_TYPE keyword' else $
if iarrays NE arrays then message,$
'ERROR - '+fname+': ITEMS count and ITEM_TYPE count discrepancy'
data_type = [temporary(data_type),item_type]
typ_ind = [temporary(typ_ind),it_ind]
endif
endif
; Remove table name from column names if present
;if nam_ind(0) LT obj_ind(1) then begin
; name = name(1:cols)
; nam_ind = nam_ind(1:cols)
;endif
; columns = strarr(cols+1)
; columns(0) = 'columns'
tabcomp=v_objpds(label,/ALL) ; 'label' is OK here
tabcomp=tabcomp[where(tabcomp.start gt object.start and tabcomp.stop lt object.stop)]
compcount=0
for i=0,size(tabcomp,/n_elem)-1 do begin
tmp=v_pdspar(label[tabcomp[i].start:tabcomp[i].stop],'NAME',count=c)
compcount+=c
endfor
tmp=v_pdspar(lbl_obj,'NAME',count=count)
if count ne compcount then begin ; TABLE has a name too
nam_ind=nam_ind[1:*]
name=name[1:*]
endif
columns = strarr(cols)
; Trim extraneous characters from column names and data_types
arch = strarr(cols)
dataTt=data_type ; for future use
for j = 0,cols-1 do begin
nmst = strpos(name(j),'"')+1 ; remove '"'s from names
if nmst GT 0 then $
name(j)=strmid(name(j),nmst,strpos(name(j),'"',nmst)-nmst)
nmst = strpos(name(j),"'")+1 ; remove "'"s from names
if nmst GT 0 then $
name(j)=strmid(name(j),nmst,strpos(name(j),"'",nmst)-nmst)
nmpar = strpos(name(j),'(') ; remove '()'s from names
if nmpar GT 0 then name(j)= strmid(name(j),0,nmpar)
nmst = strpos(name(j),10b) ; remove end-of-line controls
if nmst LT 0 then nmst = strpos(name(j),13b)
if nmst GT 0 then name(j) = strmid(name(j),0,nmst-1)
dtst = strpos(data_type(j),'"')+1 ; remove '"'s from data types
if dtst GT 0 then $
data_type(j) = strmid(data_type(j),dtst,strpos(data_type(j),'"',dtst)-dtst)
dtst = strpos(data_type(j),"'")+1 ; remove "'"s from data types
if dtst GT 0 then $
data_type(j) = strmid(data_type(j),dtst,strpos(data_type(j),"'",dtst)-dtst)
dtst = strpos(data_type(j),10b) ; remove end-of-line controls
if dtst LT 0 then dtst = strpos(data_type(j),13b)
if dtst GT 0 then data_type(j) = strmid(data_type(j),0,dtst-1)
spot = strpos(data_type(j),'_')+1
if spot GT 0 then begin ; remove prefixes from data types (for v_btabvect)
arch(j)=strmid(data_type(j),0,spot(0)-1) ; and store in 'arch'
dataTt(j)=data_type(j) ; save completer for future use (in v_typepds)
data_type(j)=strmid(data_type(j),spot,strlen(data_type(j))-spot+1)
endif
endfor
name = strtrim(name,2)
data_type = strtrim(data_type,2)
columns = name
; Read the table dimensions
X = v_pdspar( lbl_obj,'ROW_BYTES')
Y = v_pdspar( lbl_obj,'ROWS')
X = long(X(0))
Y = long(Y(0))
; ------ Read pointer to find location of the table data
; Inform user of program status if /SILENT not set
if not (SILENT) then begin
st = (cols*Y)
text = strtrim(string(cols),2)+' Columns and '+strtrim(string(Y),2)+' Rows'
if (st GT 0) then message,'Now reading table with '+text,/INFORM else begin
message,fname+" has ROWS or COLUMNS = 0, no data read", /cont
return, -1
endelse
endif
; parse pointer to data object
PtObj = V_POINTPDS(object.pointer,record_bytes)
skip = PtObj.offset ; offset in bytes
datafile_found = (PtObj.filename NE '')
if datafile_found NE 0 then begin ; if detached label, look for file location
fname = file_search(PtObj.filename, /fold) ; works from IDL 5.5 and up
temp = file_info(fname)
; If not found in current directory, try in label directory
if fname EQ "" or not(temp.exists) then begin
DirName = v_getpath(filename, FBname) ; get path to label under IDL ³ 5.4
fname = file_search(Dirname+PtObj.filename, /fold)
temp = file_info(fname)
endif
if fname EQ "" or not(temp.exists) then message, 'ERROR - Could not re-open '+ PtObj.filename
endif
; old method
;if datafile_found NE 0 then begin ; if detached label, look for file location
; works from 6.0 only
;; dir = File_DirName(fname, /mark) ; look in label directory if set
;; fname = dir + PtObj.filename
; fname = PtObj.filename
; temp = file_info(fname)
;; If the exact file name didn't work, try change case:
; if not(temp.exists) then begin
; works from IDL 5.5 only
; fname = file_search(fname, /fold)
; alternatively, look for lowcases only
;; fname = dir + strlowcase(PtObj.filename)
; temp = file_info(fname)
; endif
; if not(temp.exists) then message, 'ERROR - Could not re-open '+fname
;endif
; ------ Read data into a byte array
openr, unit, fname, ERROR = err, /GET_LUN, /compress
if err LT 0 then message,'Error opening file ' + ' ' + fname
table=bytarr(X+prefix_bytes+suffix_bytes,Y,/NOZERO)
point_lun,unit,skip
readu,unit,table
free_lun, unit
; separate prefix and suffix from data (@SE2012)
If prefix_bytes NE 0 then begin
prefix = table(0:prefix_bytes-1, *)
table = table(prefix_bytes:*, *)
endif
If suffix_bytes NE 0 then begin
suffix = table(X:*, *)
table = table(0:X-1, *)
endif
; -------Interpret correct values from byte array table
; data = CREATE_STRUCT('column_names',columns)
; SE 2012: using nam_ind fails miserably if column contain other objects (alias...) - use col_ind instead of nam_ind (?)
for k=0,cols-1 do begin
if k LT cols-1 then begin
;M.Barker 10/8/99, fixed SE 2005
;If scaling and/or offset factors are present:
if scale_flag then sf = where(sc_ind GT nam_ind(k) AND sc_ind LT nam_ind(k+1))
if offset_flag then off = where(off_ind GT nam_ind(k) AND off_ind LT nam_ind(k+1))
;----------------------------
st = where(st_ind GT nam_ind(k) AND st_ind LT nam_ind(k+1))
dt = where(typ_ind GT nam_ind(k) AND typ_ind LT nam_ind(k+1))
l = where(len_ind GT nam_ind(k) AND len_ind LT nam_ind(k+1),bitenum)
if arrays(0) GT 0 then $
it = where(is_ind GT nam_ind(k) AND is_ind LT nam_ind(k+1))
endif else begin
;M.Barker 10/8/99 This is for the last column:
if scale_flag then sf = where(sc_ind GT nam_ind(k))
if offset_flag then off = where(off_ind GT nam_ind(k))
;------------------------
st = where(st_ind GT nam_ind(k))
dt = where(typ_ind GT nam_ind(k))
l = where(len_ind GT nam_ind(k),bitenum)
if arrays(0) GT 0 then $
it = where(is_ind GT nam_ind(k))
endelse
;M.Barker 10/8/99, fixed SE 2005
if scale_flag then if sf(0) NE -1 then sf = sf(0)
if offset_flag then if off(0) NE -1 then off = off(0)
;-------------------------
st = st(0)
dt = dt(0)
elem = 1L
if arrays(0) GT 0 then if it(0) GT -1 then elem = long(items(it(0)))
; If more than one 'l' find the one that is smallest
least = l(0)
for b = 0,bitenum-1 do $
if v_str2num(length(l(b))) LT v_str2num(length(least)) then least = l(b)
l = least(0)
if st LT 0 OR dt LT 0 then message,$
'ERROR - '+fname+': column parameters missing or out of order.'
column = 'column'+strtrim(string(k+1),2)
; Extract column values
dataT = data_type(dt) ; allows modifications in routine
Flagconv = 0
vect=v_btabvect(table,col_start(st),v_str2num(length(l)),dataT,elem)
data_type(dt) = dataT
If dataT EQ 'BYTE' and length(l) EQ 6 then Flagconv = 1 ; non-std floats, stored as byte chain
; Just a test, @SE2012
; fct wants: table, Dtype, rows, bytes, [repetitions]
; ie: <data>, data_Type, ROWS, BYTES (?) - need to prepare data first, too tricky to recycle
; vect=v_btabvect2(table,data_type(dt),col_start(st),v_str2num(length(l)))
; Convert dimensions and type, swap if necessary, @SE 2005, fixed 2012
vect = reform(vect)
Col_type = v_typepds(dataTt(dt), length(l), ITYPE = integer_type, Stype = sample_type)
If flagConv EQ 1 then Col_type = 1 ; force convert to Bytes
;print, vect, ' ', k
vect = fix(vect, type=Col_type)
; print, dataTt(dt), col_type,' ', sample_type
CASE sample_type OF
'MSB': V_swapData, vect, SILENT = silent
'LSB': V_swapData, vect, /LSB, SILENT = silent
'IEEE': V_swapData, vect, SILENT = silent
'PC': V_swapData, vect, /LSB, SILENT = silent
'VAX': v_vaxtoIEEE, vect ; always floats
else: If ~SILENT then message,'WARNING - Unrecognized SAMPLE_TYPE ('$
+data_type(dt)+'), no swap performed', /INF
ENDCASE
; Convert signed bytes, not an IDL type
if (Col_type EQ 1 AND integer_type EQ 'SIGNED') then begin
; Allocate an array of 2-byte integers to hold the final values:
vect = fix(vect)
fixitlist = WHERE(vect GT 127)
if fixitlist[0] GT -1 then begin
vect[fixitlist] = vect[fixitlist] - 256
endif
endif
; Check that data type is of the right sign
; if strpos(data_type(dt),'UNSIGNED') GT -1 then vect = abs(vect)
;M.Barker 10/13/99:
;if scale and/or offset factors are present, we must perform the necessary
;operation(s) on the vector:
if scale_flag then $
if sf NE -1 then vect = vect * double(scale(sf))
if offset_flag then $
if off NE -1 then vect = vect + double(offset(off))
;-----------------------------------
vect = fix(vect, type=Col_type)
If k EQ 0 then begin
data = CREATE_STRUCT(column,vect)
endif else $
data = CREATE_STRUCT(data,column,vect)
vect = 0
endfor
; if ~SILENT then help, /STRUCTURE, data
If prefix_bytes NE 0 then data = CREATE_STRUCT (data, 'prefix',prefix) If suffix_bytes NE 0 then data = CREATE_STRUCT (data, 'suffix',suffix)
; Return data table in IDL structure form
return, data
end