forked from UCBerkeleySETI/rawspec
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrawspec.h
240 lines (199 loc) · 10.4 KB
/
rawspec.h
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
#ifndef _MYGPUSPEC_H_
#define _MYGPUSPEC_H_
#include <unistd.h>
// Specifies forward or inverse FFT direction
#define RAWSPEC_FORWARD_FFT (+1)
#define RAWSPEC_INVERSE_FFT (-1)
#define MAX_OUTPUTS (4)
#define RAWSPEC_BLOCSIZE(pctx) \
( \
((pctx)->Nc * \
(pctx)->Ntpb * \
(pctx)->Np * \
2 /* complex */ * \
(pctx)->Nbps) / 8 \
)
#define RAWSPEC_CALLBACK_PRE_DUMP (0)
#define RAWSPEC_CALLBACK_POST_DUMP (1)
// Forward declaration
typedef struct rawspec_context_s rawspec_context;
typedef void (* rawspec_dump_callback_t)(rawspec_context * ctx,
int output_product,
int callback_type);
// Structure for holding the context.
struct rawspec_context_s {
unsigned int No; // Number of output products (max MAX_OUTPUTS)
unsigned int Np; // Number of polarizations (in input data)
unsigned int Nant; // Number of antenna (coarse channels is a multiple of this)
unsigned int Nc; // Number of coarse channels
unsigned int Ntpb; // Number of time samples per block
unsigned int exit_soon; // 1:exit ASAP; 0:keep on going
// In order to better manage device memory usage, chunks of channels are
// processed at a time, instead of all channels. A value of 0 acts as a
// negative flag within the initialisation function.
unsigned int Nbc; // Number of coarse channels per batch (Batched-Channels)
// Nbps is the number of bits per sample (per component). The only supported
// values are 4* or 8 or 16. Illegal values will be treated as 8.
// 4 bits per sample (assumed to be paired as an 8bit complex byte, the most
// significant 4 bits are the real bits, the least the imaginary)
// are expanded to 8 bits on device side, see
// rawspec_copy_blocks_to_gpu_expanding_complex4.
unsigned int Nbps; // Number of bits per sample (per component)
// Npolout is the number of output polarization values per fine channel.
// This valid values for this field are:
// +1 == total power only
// +4 == full stokes mode (I, Q, U, V)
// -4 == full pol mode (XX, YY, re(XY), im(XY))
// A value of +4 or -4 is only valid if Np is 2.
// Every output product gets its own value.
int Npolout[MAX_OUTPUTS];
// Nts is an array of Nt values, one per output product. The Nt value is the
// number of time samples per FFT for a given output product. All Nt values
// must evenly divide the total number of time samples in the input buffer.
unsigned int Nts[MAX_OUTPUTS]; // Array of Nt values
// Nas is an array of Na values, one per output product. The Na value is the
// number of FFT spectra to accumulate per integration.
// Nt*Na < Nb*Ntpb : Multiple integrations per input buffer,
// must have integer integrations per input buffer
// Nt*Na = Nb*Ntpb : One integration per input buffer
// Nt*Na > Nb*Ntpb : Multiple input buffers per integration,
// must have integer input buffers per integration
unsigned int Nas[MAX_OUTPUTS]; // Array of Na values
// dump_callback is a pointer to a user-supplied output callback function.
// This function will be called twice per dump: one time just before data are
// dumped to the the output power buffer (h_pwrbuf[i]) and a second time just
// after the data are dumped to the output power buffer. The first call
// provides the client a chance to synchronize with any previous output
// thread it may have launched. The second call is when the client can
// output the data (e.g. by launching an output thread).
rawspec_dump_callback_t dump_callback;
// Pointer to user data. This is intended for use by the client's dump
// callback function (e.g. to hold output file handles or network sockets).
// The rawsepc library does not do anything with this field.
void * user_data;
// Nb is the number of input blocks per GPU input buffer.
// Set to zero to have the library calculate Nb_dev (as Ntmax/Ntpb).
unsigned int Nb;
// Nb_host is the number of input block buffers on the host.
// Set to zero to have the library calculate Nb_host (as Ntmax/Ntpb).
unsigned int Nb_host;
// Array of host pointers to Nb_host block buffers. Set h_blkbufs to NULL to
// have the library allocate the host input block buffers. For client
// managed buffers, the user must allocate an array of Nb_host pointers,
// initialize them to point to the caller allocated blocks, and set the Nb
// field of this structure. The size, in bytes, of each host input block
// buffer is Nc * Ntpb * Np * 2 * Nbps / 8.
char ** h_blkbufs;
// Which GPU to use. Set to 0 for single GPU system.
int gpu_index;
// Flag indicating that the input data are conjugated (e.g. due to frequency
// reversal/flip in the IF system). This is used on full-pol and full-stokes
// modes to ensure that the cross-pol product is conjugated correctly. Set
// this field to 0 if the frequencies are in "normal" (sky) order. Set this
// field to 1 if the frequencies are in "flipped" (reverse) order. This
// field is ignored when Npolout==1. Changes to this field take effect on
// the next call to rawspec_initialize().
int input_conjugated;
// Flag indicating the concurrent output of the output data's incoherent-sum.
int incoherently_sum;
int Naws;
// An array for the per-antenna weights to be used in the incoherent-sum.
float *Aws;
// Fields above here should be specified by client. Fields below here are
// managed by library (but can be used by the caller as needed).
// Host pointers to the output power buffers.
// In total power mode, the sizes (in bytes) will be:
// Nds[i] * Nc * Nts[i] * sizeof(float)
// In total power mode, the output buffer is [P00+P11]
// In full pol mode, the sizes (in bytes) will be:
// 4 * Nds[i] * Nc * Nts[i] * sizeof(float)
// In full pol mode, the output buffer is [P00, P11, P01re, P01im].
float * h_pwrbuf[MAX_OUTPUTS];
size_t h_pwrbuf_size[MAX_OUTPUTS];
// Host pointers to the output incoherent-sum buffers.
// This is only assigned if the appropriate execution flag is set,
// and will have sizes equal to h_pwrbuf_size[i]/Nant
float * h_icsbuf[MAX_OUTPUTS];
// Array of Nd values (number of spectra per dump)
unsigned int Nds[MAX_OUTPUTS];
// Fields below here are not normally needed at all by the client
unsigned int Ntmax; // Maximum Nt value
void * gpu_ctx; // Host pointer to opaque/private GPU specific context
};
// enum for output mode
typedef enum {
RAWSPEC_FILE,
RAWSPEC_NET
} rawspec_output_mode_t;
#ifdef __cplusplus
extern "C" {
#endif
// Returns a pointer to a string containing the librawspec version
const char * get_librawspec_version();
// Returns a pointer to a string containing the cuFFT version
const char * get_cufft_version();
// Sets ctx->Ntmax.
// Allocates host and device buffers based on the ctx->N values.
// Allocated buffers are not cleared, except for the power outbut buffers.
// Allocates and sets the ctx->rawspec_gpu_ctx field.
// Creates CuFFT plans.
// Creates streams.
// Returns 0 on success, non-zero on error.
int rawspec_initialize(rawspec_context * ctx);
// Frees host and device buffers based on the ctx->N values.
// Frees and sets the ctx->rawspec_gpu_ctx field.
// Destroys CuFFT plans.
// Destroys streams.
// Returns 0 on success, non-zero on error.
void rawspec_cleanup(rawspec_context * ctx);
// Copy `num_blocks` consecutive blocks from `ctx->h_blkbufs` to GPU input
// buffer. Starts with source block `src_idx` to destination block `dst_idx`.
// Returns 0 on success, non-zero on error.
int rawspec_copy_blocks_to_gpu(rawspec_context * ctx,
off_t src_idx, off_t dst_idx, size_t num_blocks);
// Copy `num_blocks` consecutive blocks from `ctx->h_blkbufs` to GPU input
// buffer while expanding the assumed complex4 bytes in `ctx->h_blkbufs` to
// a byte per component with a kernel on the GPU.
// Returns 0 on success, non-zero on error.
int rawspec_copy_blocks_to_gpu_expanding_complex4(
rawspec_context * ctx, off_t src_idx, off_t dst_idx, size_t num_blocks);
// Sets `num_blocks` blocks to zero in GPU input buffer, starting with block at
// `dst_idx`. If `dst_idx + num_blocks > cts->Nb`, the zeroed blocks will wrap
// to the beginning of the input buffer, but no processing will occur. Callers
// should avoid this case as it will likely not give the desired results.
// Returns 0 on success, non-zero on error.
int rawspec_zero_blocks_to_gpu(rawspec_context * ctx,
off_t dst_idx, size_t num_blocks);
// Launches FFTs of data in input buffer. Whenever an output product
// integration is complete, the power spectrum is copied to the host power
// output buffer and the user provided callback, if any, is called. This
// function returns zero on success or non-zero if an error is encountered.
//
// The direction of the FFT is determined by the fft_dir parameter. If fft_dir
// is less than or equal to zero, an inverse (aka backward) transform is
// performed, otherwise a forward transform is performed.
//
// Processing occurs asynchronously. Use `rawspec_check_for_completion` to
// see how many output products have completed or
// `rawspec_wait_for_completion` to wait for all output products to be
// complete. New data should NOT be copied to the GPU until
// `rawspec_check_for_completion` returns `ctx->No` or
// `rawspec_wait_for_completion` returns 0.
int rawspec_start_processing(rawspec_context * ctx, int fft_dir);
// Calls the appropriate rawspec_copy_blocks_to_gpu(), and then
// rawspec_start_processing.
int rawspec_copy_blocks_to_gpu_and_start_processing(rawspec_context * ctx, size_t num_blocks, char expand4bps_to8bps, int fft_dir);
// Waits for any processing to finish, then clears output power buffers and
// resets inbuf_count to 0. Returns 0 on success, non-zero on error.
int rawspec_reset_integration(rawspec_context * ctx);
// Returns the number of output products that are complete for the current
// input buffer. More precisely, it returns the number of output products that
// are no longer processing (or never were processing) the input buffer.
unsigned int rawspec_check_for_completion(rawspec_context * ctx);
// Waits for any pending output products to be compete processing the current
// input buffer. Returns zero when complete, non-zero on error.
int rawspec_wait_for_completion(rawspec_context * ctx);
#ifdef __cplusplus
}
#endif
#endif // _MYGPUSPEC_H_