;+ ;procedure: thm_cal_fgm ; ;Purpose: applies calibration to THEMIS fluxgate magnetometer data ; ; searches for the current calibration in ASCII file ; takes calibration that has time stamp before or at the start of the data ; converts data to nano Tessla ; applies the calibration ; corrects for phase shift from averaging before despinning ; ;keywords: ; probe = Probe name. The default is 'all', i.e., calibrate data for all ; available probes. ; This can be an array of strings, e.g., ['a', 'b'] or a ; single string delimited by spaces, e.g., 'a b' ; datatype = The type of data to be loaded, 'fge', 'fgh', or 'fgl'. default is ; 'all', to calibrate all variables. ; in_suffix = optional suffix to add to name of input data quantity, which ; is generated from probe and datatype keywords. ; out_suffix = optional suffix to add to name for output tplot quantity, ; which is generated from probe and datatype keywords. ; coord = apply thm_cotrans if output other ; than spinning spacecraft (SSL) is desired. ; ;optional parameters: ; ; name_thx_fgx_in --> input data (t-plot variable name) ; name_thx_fgx_hed --> header information for input data (t-plot variable name) ; name_thx_fgx_out --> name for output (t-plot variable name) ; pathfile --> path and filename of the calibration file ; ;keywords: ;Example: ; tha_cal_fgm, probe = 'a', datatype= 'fgl' ; ;Notes: under construction!! ; ;Written by Hannes Schwarzl. ; $LastChangedBy: jimm $ ; $LastChangedDate: 2007-09-27 11:40:03 -0700 (Thu, 27 Sep 2007) $ ; $LastChangedRevision: 1634 $ ; $URL: svn+ssh://thmsvn@ambrosia.ssl.berkeley.edu/repos/thmsoc/tags/tdas_2_02/idl/themis/spacecraft/fields/thm_cal_fgm.pro $ ;Changes by Edita Georgescu ;eg 6/3/2007 - matrix multiplication ;eg 30/3/2007 - keyword deleted, fgmtype determined from: name_thx_fgx_in ; - calibrates fgl,fgh,fge ; - calibrate only mode=0 data ; 2007-03-30 Modified by Hannnes S to use multiple calibrations for a selected interval ; and use spinperiod from the statefile ; Search this string to see changes ;Hannes 30/3/2007 ; 2007-03-30 Modified by Hannnes S correction for filter delay removed ; Search this string to see changes: Hannes 05/21/2007 ;- pro thm_cal_fgm,probe=probe, datatype=datatype, in_suffix=in_suf, out_suffix=out_suf, coord=coord, name_thx_fgx_in,name_thx_fgx_hed,name_thx_fgx_out,pathfile ;eg 30/3/2007;Hannes 30/3/2007 ;kb ;; implement the 'standard' interface as a wrapper around the original ;; interface if no positional parameters are present. if n_params() eq 0 then begin vprobes = ['a','b','c','d','e'] vdatatypes = ['fgl', 'fgh', 'fge'] if keyword_set(valid_names) then begin probe = vprobes datatypes = vdatatypes return endif if not keyword_set(probe) then probes = vprobes $ else probes = thm_check_valid_name(strlowcase(probe), vprobes, /include_all) if not keyword_set(probes) then return if not keyword_set(datatype) then dts = vdatatypes $ else dts = thm_check_valid_name(strlowcase(datatype), vdatatypes, $ /include_all) if not keyword_set(dts) then return if not keyword_set(in_suf) then in_suf = '' if not keyword_set(out_suf) then out_suf = '' for i = 0, n_elements(probes)-1 do begin thx = 'th' + probes[i] cal_relpathname = thx+'/l1/fgm/0000/'+thx+'_fgmcal.txt' cal_file = file_retrieve(cal_relpathname, _extra=!themis) for j = 0, n_elements(dts)-1 do begin raw_name = 'th'+probes[i]+'_'+dts[j]+in_suf cal_name = 'th'+probes[i]+'_'+dts[j]+out_suf hed_name = 'th'+probes[i]+'_'+dts[j]+'_hed' thm_cal_fgm, raw_name, hed_name, cal_name, cal_file, coord=coord endfor endfor return endif else if n_params() ne 4 then begin print, 'usage: thm_cal_fgm, probe=probe, datatype=datatype $' print, ' [, in_suffix=in_suf, out_suffix=out_suf]' print, 'or: thm_cal_fgm, raw_name, hed_name, cal_name, cal_file]' return endif ; what follows is exactly the original thm_cal_fgm, before 'standardization' ; (plus some metadata in dlimit.data_att) ;kb ; get the data using t-plot names get_data,name_thx_fgx_in,data=thx_fgx_in,dl=thx_fgx_dl ;kb get_data,name_thx_fgx_hed,data=thx_fgx_hed ; kb if size(thx_fgx_in,/type) ne 8 then begin print, name_thx_fgx_in, " does not exist." return endif if size(thx_fgx_hed,/type) ne 8 then begin print, name_thx_fgx_hed, " does not exist: " message, " Support data not found." endif if thm_data_calibrated(thx_fgx_dl) then begin message, name_thx_fgx_in + " has already been calibrated." endif ; end kb ; make STRUCTRE thx_fgx; converting egineering units from LONG to DOUBLE thx_fgx = CREATE_STRUCT('X',thx_fgx_in.X , 'Y', DOUBLE(thx_fgx_in.Y)) ;, 'V', thx_fgx_in.V) -mattd,5/2 ; scale factor kr=2.980232238769531E-3 ;kr=25000.0/2^23 --> see *CALPROC*.doc ; fillvalue for output vector ;fillvalue=1.e-31 fillvalue=!values.f_nan ;Hannes 30/3/2007 ; check input ;eg 30/3/2007 fgt=['fgl','fgh','fge'] fgtype=strmid(name_thx_fgx_in,strpos(name_thx_fgx_in,'_fg')+1,3) tm=where(fgtype eq fgt) & print,name_thx_fgx_in ;eg 30/3/2007 ; get array sizes count=SIZE(thx_fgx.X,/N_ELEMENTS) countHed=SIZE(thx_fgx_HED.X,/N_ELEMENTS) PRINT,'get header information ...' ; get the information from the header fgmhed=thx_fgx_HED.Y[*,12:15] ; store relevant header information PRINT,'get sampling frequencies ...' ; get fs [vector] ;eg 30/3/2007 case tm of 0 : fs=float(2^(2+(fgmhed[*,3] and '111'B))) 1 : fs=replicate(128.0,countHed) 2 : fs=replicate(8.0,countHed) endcase PRINT,'get filter-modes ...' ; get filtermode [vector] if tm lt 2 then ff=ishft('F0'X and fgmhed[*,1],-6) else ff=replicate(0,countHed) ; get mode [vector] only mode=0 is science data and have to be calibrated ;eg 30/3/2007 if tm lt 2 then mode=ishft(fgmhed[*,1] and '1100'B,-2) else mode=replicate(0,countHed) ;eg 30/3/2007 PRINT,'done getting header information' ;read the calibration file PRINT,'read calibration file:' PRINT,pathfile ncal=file_lines(pathfile) ;define variables calstr=strarr(ncal) spinperi=dblarr(ncal) offi=dblarr(ncal,3) cali=dblarr(ncal,9) spinperii=dblarr(1) offii=dblarr(3) calii=dblarr(9) utci='2006-01-01T00:00:00.000Z' utc=dblarr(ncal) utcStr=strarr(ncal) openr,2,pathfile readf,2,calstr close,2 for i=0,ncal-1 DO BEGIN calstri=calstr(i) utci=strmid(calstr(i),0,25) ;eg 6/3/2007 reads,strmid(calstr(i),26),offii,calii,spinperii;,format="(a25,3f5,9f9,f9)" ;eg 6/3/2007 offi(i,*)=offii cali(i,*)=calii spinperi(i)=spinperii utcStr(i)=utci ;translate time information STRPUT, utci, '/', 10 utc(i)=time_double(utci) ENDFOR PRINT,'done reading calibration file' ;Hannes 30/3/2007 ;get the spinperiod from the state file preSpinper=strmid(name_thx_fgx_in,0,4) name_thx_spinper=preSpinper+'state_spinper' PRINT,'take spin period from state file ...' get_data,name_thx_spinper,data=thx_spinper if size(thx_spinper,/type) ne 8 then begin message, 'aborted: must load spin data from state file' endif thx_spinper_interp=thm_interpolate_state(thx_xxx_in=thx_fgx_in,thx_spinper=thx_spinper) ;--> linear interpolation ;end Hannes 30/3/2007 PRINT,'search calibration for selected time interval ...' calIndex=0 compTime=utc(0) refTime=thx_fgx.X(0) i=0 WHILE ((compTime lt reftime) && (i lt ncal-1)) DO BEGIN i=i+1 compTime=utc(i) IF (compTime gt reftime) THEN BEGIN i=i-1 BREAK ENDIF ENDWHILE ;change 2007-03-29 istart=i compTime=utc(i) refTime=thx_fgx.X(count-1L) ;i=0 WHILE ((compTime lt reftime) && (i lt ncal-1)) DO BEGIN i=i+1 compTime=utc(i) IF (compTime gt reftime) THEN BEGIN i=i-1 BREAK ENDIF ENDWHILE istop=i PRINT, 'Select calibrations from:' FOR i=istart,istop DO BEGIN PRINT, utcStr(i) ENDFOR ;PRINT,'offsets: ',TRANSPOSE(offs) ;PRINT,'cal matrix: ' ;PRINT,calm ;PRINT,'spin period: ',spinper ;PRINT, 'Select calibration from:' ;PRINT, utcStr(i) ;offs=offi(i,*) ;calim=TRANSPOSE(cali(i,*)) ;calm=[[calim[0:2]],[calim[3:5]],[calim[6:8]]] ;spinper=spinperi(i) Hannes 30/3/2007 ;if abs(spinper-mean(tspin.y))/spinper gt 1.e-4 then spinper=mean(tspin.y) ;eg 30/3/2007,Hannes 30/3/2007 ;PRINT,'offsets: ',TRANSPOSE(offs) ;PRINT,'cal matrix: ' ;PRINT,calm ;PRINT,'spin period: ',spinper Hannes 30/3/2007 ;end Hannes 30/3/2007 PRINT,'apply scale factor ...' ;apply scale factor thx_fgx.Y=thx_fgx.Y*kr PRINT,'apply calibration ...' ;apply calibration matrix and offset ;lastfs=0 i=0L ffi=ff[i] fsi=fs[i] modi=mode(i) ;eg 30/3/2007 timei=thx_fgx_HED.X[i] ;Hannes 30/3/2007 ;starting calibration offs=offi(istart,*) calim=TRANSPOSE(cali(istart,*)) calm=[[calim[0:2]],[calim[3:5]],[calim[6:8]]] ;spinper=spinperi(istart) iact=istart IF (iact eq istop) THEN BEGIN nextCalChangeTime=thx_fgx.X(count-1L)+10.d0 ;something greater than the last point in time ENDIF ELSE BEGIN nextCalChangeTime=utc(iact+1L) PRINT, 'nextCalChangeTime:' PRINT, utcStr(iact+1L) ENDELSE ;end Hannes 30/3/2007 for j=0L,count-1L do begin ;Hannes 30/3/2007 ;apply the right calibration at the right time IF (thx_fgx.X[j] ge nextCalChangeTime) THEN BEGIN iact=iact+1 IF (iact eq istop) THEN BEGIN nextCalChangeTime=thx_fgx.X(count-1L)+10.d0 ;something greater than the last point in time ENDIF ELSE BEGIN nextCalChangeTime=utc(iact+1L) PRINT, 'nextCalChangeTime:' PRINT, utcStr(iact+1L) ENDELSE offs=offi(iact,*) calim=TRANSPOSE(cali(iact,*)) calm=[[calim[0:2]],[calim[3:5]],[calim[6:8]]] ;spinper=spinperi(iact) ENDIF ;end Hannes 30/3/2007 ; update header information if necessary (WHILE loop should work for all possible cases) while ((thx_fgx.X[j] gt timei) && (i lt countHed-1)) do begin i=i+1 ffi=ff[i] fsi=fs[i] modi=mode(i) timei=thx_fgx_HED.X[i] endwhile if modi eq 0 then begin ;eg 30/3/2007 ; thx_fgx.Y[j,*]=MATRIX_MULTIPLY(calm, thx_fgx.Y[j,*], /BTRANSPOSE)-offs ;eg 6/3/2007 thx_fgx.Y[j,*]=calm ## thx_fgx.Y[j,*] - offs ;eg 6/3/2007 ;correct for filter if ffi eq 2 then begin spinper=thx_spinper_interp.Y[j];Hannes 30/3/2007 arg=-!dpi/fsi/spinper dfilt=128./fsi*sin(!dpi/128.d0/spinper)/sin(-arg) dfm=identity(3,/double) dfm[1,1]=dfilt dfm[0,0]=dfilt ;sarg=sin(arg) /delay removed Hannes 05/21/2007 ;carg=cos(arg) ;& print,arg,dfilt /delay removed Hannes 05/21/2007 ;delaym=identity(3,/double) /delay removed Hannes 05/21/2007 ;delaym[0,1]=-sarg ;eg 6/3/2007 /delay removed Hannes 05/21/2007 ;delaym[1,0]=sarg ;eg 6/3/2007 /delay removed Hannes 05/21/2007 ;delaym[0,0]=carg /delay removed Hannes 05/21/2007 ;delaym[1,1]=carg /delay removed Hannes 05/21/2007 ;mfilt=dfm#delaym /delay removed Hannes 05/21/2007 mfilt=dfm ;/apply only amplitude correction Hannes 05/21/2007 ; thx_fgx.Y[j,*]=MATRIX_MULTIPLY(mfilt, thx_fgx.Y[j,*],/BTRANSPOSE) ;eg 6/3/2007 thx_fgx.Y[j,*]=mfilt ## thx_fgx.Y[j,*] ;eg 6/3/2007 endif ;correct for filter endif else thx_fgx.Y[j,*]=fillvalue ;eg 30/3/2007 set vectors to fillvalue ??? endfor PRINT,'calibration applied' PRINT,'done' ;store data, where mode = 0 ; kb units = 'nT' dl = thx_fgx_dl str_element, dl, 'data_att', data_att, success=has_data_att if has_data_att then begin str_element, data_att, 'data_type', 'calibrated', /add endif else data_att = { data_type: 'calibrated' } str_element, data_att, 'coord_sys', 'ssl', /add str_element, data_att, 'units', units, /add str_element, dl, 'data_att', data_att, /add str_element, dl, 'ytitle', string( name_thx_fgx_out, units, format='(A,"!C!C[",A,"]")'), /add str_element, dl, 'labels', [ 'x', 'y', 'z'], /add str_element, dl, 'labflag', 1, /add str_element, dl, 'colors', [ 2, 4, 6], /add str_element, dl, 'color_table', 39, /add ; end kb store_data,name_thx_fgx_out,data=thx_fgx,dl=dl ; kb if keyword_set(coord) && strlowcase(coord) ne 'ssl' then begin thm_cotrans, name_thx_fgx_out, out_coord = coord options, tplot_var, 'ytitle', /def, $ string( name_thx_fgx_out, units, format='(A,"!C!C[",A,"]")'), /add endif ; end kb ;return,thx_fgx end