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
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
|
/* -*- c++ -*- */
/*
* Copyright 2006 Free Software Foundation, Inc.
*
* This file is part of GNU Radio
*
* GNU Radio is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* GNU Radio is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with GNU Radio; see the file COPYING. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street,
* Boston, MA 02110-1301, USA.
*/
#ifndef INCLUDED_PMT_H
#define INCLUDED_PMT_H
#include <boost/shared_ptr.hpp>
#include <complex>
#include <string>
#include <stdint.h>
#include <iostream>
/*!
* This file defines a polymorphic type and the operations on it.
*
* It draws heavily on the idea of scheme and lisp data types.
* The interface parallels that in Guile 1.8, with the notable
* exception that these objects are transparently reference counted.
*/
/*!
* \brief base class of all pmt types
*/
class pmt_base;
/*!
* \brief typedef for shared pointer (transparent reference counting).
* See http://www.boost.org/libs/smart_ptr/smart_ptr.htm
*/
typedef boost::shared_ptr<pmt_base> pmt_t;
class pmt_exception
{
const char *d_msg;
pmt_t d_obj;
public:
pmt_exception(const char *msg, pmt_t obj);
const char *msg() { return d_msg; }
pmt_t obj() { return d_obj; }
};
class pmt_wrong_type : public pmt_exception
{
public:
pmt_wrong_type(const char *msg, pmt_t obj);
};
class pmt_out_of_range : public pmt_exception
{
public:
pmt_out_of_range(const char *msg, pmt_t obj);
};
/*
* ------------------------------------------------------------------------
* Booleans. Two constants, #t and #f.
*
* In predicates, anything that is not #f is considered true.
* I.e., there is a single false value, #f.
* ------------------------------------------------------------------------
*/
extern const pmt_t PMT_BOOL_T; //< #t : boolean true constant
extern const pmt_t PMT_BOOL_F; //< #f : boolean false constant
//! Return true if obj is #t or #f, else return false.
bool pmt_is_bool(pmt_t obj);
//! Return false if obj is #f, else return true.
bool pmt_is_true(pmt_t obj);
//! Return true if obj is #f, else return true.
bool pmt_is_false(pmt_t obj);
//! Return #f is val is false, else return #t.
pmt_t pmt_from_bool(bool val);
//! Return true if val is PMT_BOOL_T, return false when val is PMT_BOOL_F,
// else raise wrong_type exception.
bool pmt_to_bool(pmt_t val);
/*
* ------------------------------------------------------------------------
* Symbols
* ------------------------------------------------------------------------
*/
//! Return true if obj is a symbol, else false.
bool pmt_is_symbol(pmt_t obj);
//! Return the symbol whose name is \p s.
pmt_t pmt_string_to_symbol(const std::string &s);
/*!
* If \p is a symbol, return the name of the symbol as a string.
* Otherwise, raise the wrong_type exception.
*/
const std::string pmt_symbol_to_string(pmt_t sym);
/*
* ------------------------------------------------------------------------
* Numbers: we support integer, real and complex
* ------------------------------------------------------------------------
*/
//! Return true if obj is any kind of number, else false.
bool pmt_is_number(pmt_t obj);
/*
* ------------------------------------------------------------------------
* Integers
* ------------------------------------------------------------------------
*/
//! Return true if \p x is an integer number, else false
bool pmt_is_integer(pmt_t x);
//! Return the pmt value that represents the integer \p x.
pmt_t pmt_from_long(long x);
/*!
* \brief Convert pmt to long if possible.
*
* When \p x represents an exact integer that fits in a long,
* return that integer. Else raise an exception, either wrong_type
* when x is not an exact integer, or out_of_range when it doesn't fit.
*/
long pmt_to_long(pmt_t x);
/*
* ------------------------------------------------------------------------
* Reals
* ------------------------------------------------------------------------
*/
/*
* \brief Return true if \p obj is a real number, else false.
*/
bool pmt_is_real(pmt_t obj);
//! Return the pmt value that represents double \p x.
pmt_t pmt_from_double(double x);
/*!
* \brief Convert pmt to double if possible.
*
* Returns the number closest to \p val that is representable
* as a double. The argument \p val must be a real or integer, otherwise
* a wrong_type exception is raised.
*/
double pmt_to_double(pmt_t x);
/*
* ------------------------------------------------------------------------
* Complex
* ------------------------------------------------------------------------
*/
/*!
* \brief return true if \p obj is a complex number, false otherwise.
*/
bool pmt_is_complex(pmt_t obj);
//! Return a complex number constructed of the given real and imaginary parts.
pmt_t pmt_make_rectangular(double re, double im);
/*!
* If \p z is complex, real or integer, return the closest complex<double>.
* Otherwise, raise the wrong_type exception.
*/
std::complex<double> pmt_to_complex(pmt_t z);
/*
* ------------------------------------------------------------------------
* Pairs
* ------------------------------------------------------------------------
*/
extern const pmt_t PMT_NIL; //< the empty list
//! Return true if \p x is the empty list, otherwise return false.
bool pmt_is_null(pmt_t x);
//! Return true if \p obj is a pair, else false.
bool pmt_is_pair(pmt_t obj);
//! Return a newly allocated pair whose car is \p x and whose cdr is \p y.
pmt_t pmt_cons(pmt_t x, pmt_t y);
//! If \p pair is a pair, return the car of the \p pair, otherwise raise wrong_type.
pmt_t pmt_car(pmt_t pair);
//! If \p pair is a pair, return the cdr of the \p pair, otherwise raise wrong_type.
pmt_t pmt_cdr(pmt_t pair);
//! Stores \p value in the car field of \p pair.
void pmt_set_car(pmt_t pair, pmt_t value);
//! Stores \p value in the cdr field of \p pair.
void pmt_set_cdr(pmt_t pair, pmt_t value);
/*
* ------------------------------------------------------------------------
* Vectors
*
* These vectors can hold any kind of objects. Indexing is zero based.
* ------------------------------------------------------------------------
*/
//! Return true if \p x is a vector, othewise false.
bool pmt_is_vector(pmt_t x);
//! Make a vector of length \p k, with initial values set to \p fill
pmt_t pmt_make_vector(size_t k, pmt_t fill);
/*!
* Return the contents of position \p k of \p vector.
* \p k must be a valid index of \p vector.
*/
pmt_t pmt_vector_ref(pmt_t vector, size_t k);
//! Store \p obj in position \p k.
void pmt_vector_set(pmt_t vector, size_t k, pmt_t obj);
//! Store \p fill in every position of \p vector
void pmt_vector_fill(pmt_t vector, pmt_t fill);
/*!
* <pre>
* ------------------------------------------------------------------------
* Uniform Numeric Vectors
*
* A uniform numeric vector is a vector whose elements are all of single
* numeric type. pmt offers uniform numeric vectors for signed and
* unsigned 8-bit, 16-bit, 32-bit, and 64-bit integers, two sizes of
* floating point values, and complex floating-point numbers of these
* two sizes. Indexing is zero based.
*
* The names of the functions include these tags in their names:
*
* u8 unsigned 8-bit integers
* s8 signed 8-bit integers
* u16 unsigned 16-bit integers
* s16 signed 16-bit integers
* u32 unsigned 32-bit integers
* s32 signed 32-bit integers
* u64 unsigned 64-bit integers
* s64 signed 64-bit integers
* f32 the C++ type float
* f64 the C++ type double
* c32 the C++ type complex<float>
* c64 the C++ type complex<double>
* ------------------------------------------------------------------------
* </pre>
*/
//! true if \p x is any kind of uniform numeric vector
bool pmt_is_uniform_vector(pmt_t x);
bool pmt_is_u8vector(pmt_t x);
bool pmt_is_s8vector(pmt_t x);
bool pmt_is_u16vector(pmt_t x);
bool pmt_is_s16vector(pmt_t x);
bool pmt_is_u32vector(pmt_t x);
bool pmt_is_s32vector(pmt_t x);
bool pmt_is_u64vector(pmt_t x);
bool pmt_is_s64vector(pmt_t x);
bool pmt_is_f32vector(pmt_t x);
bool pmt_is_f64vector(pmt_t x);
bool pmt_is_c32vector(pmt_t x);
bool pmt_is_c64vector(pmt_t x);
pmt_t pmt_make_u8vector(size_t k, uint8_t fill);
pmt_t pmt_make_s8vector(size_t k, int8_t fill);
pmt_t pmt_make_u16vector(size_t k, uint16_t fill);
pmt_t pmt_make_s16vector(size_t k, int16_t fill);
pmt_t pmt_make_u32vector(size_t k, uint32_t fill);
pmt_t pmt_make_s32vector(size_t k, int32_t fill);
pmt_t pmt_make_u64vector(size_t k, uint64_t fill);
pmt_t pmt_make_s64vector(size_t k, int64_t fill);
pmt_t pmt_make_f32vector(size_t k, float fill);
pmt_t pmt_make_f64vector(size_t k, double fill);
pmt_t pmt_make_c32vector(size_t k, std::complex<float> fill);
pmt_t pmt_make_c64vector(size_t k, std::complex<double> fill);
pmt_t pmt_init_u8vector(size_t k, const uint8_t *data);
pmt_t pmt_init_s8vector(size_t k, const int8_t *data);
pmt_t pmt_init_u16vector(size_t k, const uint16_t *data);
pmt_t pmt_init_s16vector(size_t k, const int16_t *data);
pmt_t pmt_init_u32vector(size_t k, const uint32_t *data);
pmt_t pmt_init_s32vector(size_t k, const int32_t *data);
pmt_t pmt_init_u64vector(size_t k, const uint64_t *data);
pmt_t pmt_init_s64vector(size_t k, const int64_t *data);
pmt_t pmt_init_f32vector(size_t k, const float *data);
pmt_t pmt_init_f64vector(size_t k, const double *data);
pmt_t pmt_init_c32vector(size_t k, const std::complex<float> *data);
pmt_t pmt_init_c64vector(size_t k, const std::complex<double> *data);
uint8_t pmt_u8vector_ref(pmt_t v, size_t k);
int8_t pmt_s8vector_ref(pmt_t v, size_t k);
uint16_t pmt_u16vector_ref(pmt_t v, size_t k);
int16_t pmt_s16vector_ref(pmt_t v, size_t k);
uint32_t pmt_u32vector_ref(pmt_t v, size_t k);
int32_t pmt_s32vector_ref(pmt_t v, size_t k);
uint64_t pmt_u64vector_ref(pmt_t v, size_t k);
int64_t pmt_s64vector_ref(pmt_t v, size_t k);
float pmt_f32vector_ref(pmt_t v, size_t k);
double pmt_f64vector_ref(pmt_t v, size_t k);
std::complex<float> pmt_c32vector_ref(pmt_t v, size_t k);
std::complex<double> pmt_c64vector_ref(pmt_t v, size_t k);
void pmt_u8vector_set(pmt_t v, size_t k, uint8_t x); //< v[k] = x
void pmt_s8vector_set(pmt_t v, size_t k, int8_t x);
void pmt_u16vector_set(pmt_t v, size_t k, uint16_t x);
void pmt_s16vector_set(pmt_t v, size_t k, int16_t x);
void pmt_u32vector_set(pmt_t v, size_t k, uint32_t x);
void pmt_s32vector_set(pmt_t v, size_t k, int32_t x);
void pmt_u64vector_set(pmt_t v, size_t k, uint64_t x);
void pmt_s64vector_set(pmt_t v, size_t k, int64_t x);
void pmt_f32vector_set(pmt_t v, size_t k, float x);
void pmt_f64vector_set(pmt_t v, size_t k, double x);
void pmt_c32vector_set(pmt_t v, size_t k, std::complex<float> x);
void pmt_c64vector_set(pmt_t v, size_t k, std::complex<double> x);
// Return const pointers to the elements
const void *pmt_uniform_vector_elements(pmt_t v, size_t &len); //< works with any; len is in bytes
const uint8_t *pmt_u8vector_elements(pmt_t v, size_t &len); //< len is in elements
const int8_t *pmt_s8vector_elements(pmt_t v, size_t &len); //< len is in elements
const uint16_t *pmt_u16vector_elements(pmt_t v, size_t &len); //< len is in elements
const int16_t *pmt_s16vector_elements(pmt_t v, size_t &len); //< len is in elements
const uint32_t *pmt_u32vector_elements(pmt_t v, size_t &len); //< len is in elements
const int32_t *pmt_s32vector_elements(pmt_t v, size_t &len); //< len is in elements
const uint64_t *pmt_u64vector_elements(pmt_t v, size_t &len); //< len is in elements
const int64_t *pmt_s64vector_elements(pmt_t v, size_t &len); //< len is in elements
const float *pmt_f32vector_elements(pmt_t v, size_t &len); //< len is in elements
const double *pmt_f64vector_elements(pmt_t v, size_t &len); //< len is in elements
const std::complex<float> *pmt_c32vector_elements(pmt_t v, size_t &len); //< len is in elements
const std::complex<double> *pmt_c64vector_elements(pmt_t v, size_t &len); //< len is in elements
// Return non-const pointers to the elements
void *pmt_uniform_vector_writeable_elements(pmt_t v, size_t &len); //< works with any; len is in bytes
uint8_t *pmt_u8vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
int8_t *pmt_s8vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
uint16_t *pmt_u16vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
int16_t *pmt_s16vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
uint32_t *pmt_u32vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
int32_t *pmt_s32vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
uint64_t *pmt_u64vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
int64_t *pmt_s64vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
float *pmt_f32vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
double *pmt_f64vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
std::complex<float> *pmt_c32vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
std::complex<double> *pmt_c64vector_writeable_elements(pmt_t v, size_t &len); //< len is in elements
/*
* ------------------------------------------------------------------------
* Dictionary (a.k.a associative array, hash, map)
* ------------------------------------------------------------------------
*/
//! Return true if \p obj is a dictionary
bool pmt_is_dict(pmt_t obj);
//! make an empty dictionary
pmt_t pmt_make_dict();
//! dict[key] = value
void pmt_dict_set(pmt_t dict, pmt_t key, pmt_t value);
//! Return true if \p key exists in \p dict
bool pmt_dict_has_key(pmt_t dict, pmt_t key);
//! If \p key exists in \p dict, return associated value; otherwise return \p not_found.
pmt_t pmt_dict_ref(pmt_t dict, pmt_t key, pmt_t not_found);
//! Return list of (key . value) pairs
pmt_t pmt_dict_items(pmt_t dict);
//! Return list of keys
pmt_t pmt_dict_keys(pmt_t dict);
//! Return list of values
pmt_t pmt_dict_values(pmt_t dict);
/*
* ------------------------------------------------------------------------
* General functions
* ------------------------------------------------------------------------
*/
//! Return true if x and y are the same object; otherwise return false.
bool pmt_eq(pmt_t x, pmt_t y);
/*!
* \brief Return true if x and y should normally be regarded as the same object, else false.
*
* <pre>
* eqv returns true if:
* x and y are the same object.
* x and y are both #t or both #f.
* x and y are both symbols and their names are the same.
* x and y are both numbers, and are numerically equal.
* x and y are both the empty list (nil).
* x and y are pairs or vectors that denote same location in store.
* </pre>
*/
bool pmt_eqv(pmt_t x, pmt_t y);
/*!
* pmt_equal recursively compares the contents of pairs and vectors,
* applying pmt_eqv on other objects such as numbers and symbols.
* pmt_equal may fail to terminate if its arguments are circular data
* structures.
*/
bool pmt_equal(pmt_t x, pmt_t y);
//! Return the number of elements in v
size_t pmt_length(pmt_t v);
/*!
* \brief Find the first pair in \p alist whose car field is \p obj
* and return that pair.
*
* \p alist (for "association list") must be a list of pairs. If no pair
* in \p alist has \p obj as its car then #f is returned.
* Uses pmt_eq to compare \p obj with car fields of the pairs in \p alist.
*/
pmt_t pmt_assq(pmt_t obj, pmt_t alist);
/*!
* \brief Find the first pair in \p alist whose car field is \p obj
* and return that pair.
*
* \p alist (for "association list") must be a list of pairs. If no pair
* in \p alist has \p obj as its car then #f is returned.
* Uses pmt_eqv to compare \p obj with car fields of the pairs in \p alist.
*/
pmt_t pmt_assv(pmt_t obj, pmt_t alist);
/*!
* \brief Find the first pair in \p alist whose car field is \p obj
* and return that pair.
*
* \p alist (for "association list") must be a list of pairs. If no pair
* in \p alist has \p obj as its car then #f is returned.
* Uses pmt_equal to compare \p obj with car fields of the pairs in \p alist.
*/
pmt_t pmt_assoc(pmt_t obj, pmt_t alist);
/*!
* \brief Apply \p proc element-wise to the elements of list and returns
* a list of the results, in order.
*
* \p list must be a list. The dynamic order in which \p proc is
* applied to the elements of \p list is unspecified.
*/
pmt_t pmt_map(pmt_t proc(pmt_t), pmt_t list);
/*!
* \brief reverse \p list.
*
* \p list must be a proper list.
*/
pmt_t pmt_reverse(pmt_t list);
/*!
* \brief destructively reverse \p list.
*
* \p list must be a proper list.
*/
pmt_t pmt_reverse_x(pmt_t list);
/*!
* \brief (acons x y a) == (cons (cons x y) a)
*/
inline static pmt_t
pmt_acons(pmt_t x, pmt_t y, pmt_t a)
{
return pmt_cons(pmt_cons(x, y), a);
}
/*
* ------------------------------------------------------------------------
* read / write
* ------------------------------------------------------------------------
*/
extern const pmt_t PMT_EOF; //< The end of file object
//! return true if obj is the EOF object, otherwise return false.
bool pmt_is_eof_object(pmt_t obj);
/*!
* read converts external representations of pmt objects into the
* objects themselves. Read returns the next object parsable from
* the given input port, updating port to point to the first
* character past the end of the external representation of the
* object.
*
* If an end of file is encountered in the input before any
* characters are found that can begin an object, then an end of file
* object is returned. The port remains open, and further attempts
* to read will also return an end of file object. If an end of file
* is encountered after the beginning of an object's external
* representation, but the external representation is incomplete and
* therefore not parsable, an error is signaled.
*/
pmt_t pmt_read(std::istream &port);
/*!
* Write a written representation of \p obj to the given \p port.
*/
void pmt_write(pmt_t obj, std::ostream &port);
/*
* ------------------------------------------------------------------------
* portable byte stream representation
* ------------------------------------------------------------------------
*/
/*!
* \brief Write portable byte-serial representation of \p obj to \p sink
*/
void pmt_serialize(pmt_t obj, std::ostream &sink);
/*!
* \brief Create obj from portable byte-serial representation
*/
pmt_t pmt_deserialize(std::istream &source);
#endif /* INCLUDED_PMT_H */
|