/* -*- c++ -*- */ /* * Copyright 2007,2009 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 3, 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 this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include "pmt_int.h" #include "gruel/pmt_serial_tags.h" namespace pmt { static pmt_t parse_pair(std::streambuf &sb); // ---------------------------------------------------------------- // output primitives // ---------------------------------------------------------------- static bool serialize_untagged_u8(unsigned int i, std::streambuf &sb) { return sb.sputc((i >> 0) & 0xff) != std::streambuf::traits_type::eof(); } // always writes big-endian static bool serialize_untagged_u16(unsigned int i, std::streambuf &sb) { sb.sputc((i >> 8) & 0xff); return sb.sputc((i >> 0) & 0xff) != std::streambuf::traits_type::eof(); } // always writes big-endian static bool serialize_untagged_u32(unsigned int i, std::streambuf &sb) { sb.sputc((i >> 24) & 0xff); sb.sputc((i >> 16) & 0xff); sb.sputc((i >> 8) & 0xff); return sb.sputc((i >> 0) & 0xff) != std::streambuf::traits_type::eof(); } static bool serialize_untagged_f64(double i, std::streambuf &sb) { typedef union { double id; uint64_t ii; } iu_t; iu_t iu; iu.id = i; sb.sputc((iu.ii >> 56) & 0xff); sb.sputc((iu.ii >> 48) & 0xff); sb.sputc((iu.ii >> 40) & 0xff); sb.sputc((iu.ii >> 32) & 0xff); sb.sputc((iu.ii >> 24) & 0xff); sb.sputc((iu.ii >> 16) & 0xff); sb.sputc((iu.ii >> 8) & 0xff); return sb.sputc((iu.ii >> 0) & 0xff) != std::streambuf::traits_type::eof(); } // always writes big-endian static bool serialize_untagged_u64(uint64_t i, std::streambuf &sb) { sb.sputc((i >> 56) & 0xff); sb.sputc((i >> 48) & 0xff); sb.sputc((i >> 40) & 0xff); sb.sputc((i >> 32) & 0xff); sb.sputc((i >> 24) & 0xff); sb.sputc((i >> 16) & 0xff); sb.sputc((i >> 8) & 0xff); return sb.sputc((i >> 0) & 0xff) != std::streambuf::traits_type::eof(); } // ---------------------------------------------------------------- // input primitives // ---------------------------------------------------------------- // always reads big-endian static bool deserialize_untagged_u8(uint8_t *ip, std::streambuf &sb) { std::streambuf::traits_type::int_type t; int i; t = sb.sbumpc(); i = t & 0xff; *ip = i; return t != std::streambuf::traits_type::eof(); } // always reads big-endian static bool deserialize_untagged_u16(uint16_t *ip, std::streambuf &sb) { std::streambuf::traits_type::int_type t; int i; t = sb.sbumpc(); i = t & 0xff; t = sb.sbumpc(); i = (i << 8) | (t & 0xff); *ip = i; return t != std::streambuf::traits_type::eof(); } // always reads big-endian static bool deserialize_untagged_u32(uint32_t *ip, std::streambuf &sb) { std::streambuf::traits_type::int_type t; int i; t = sb.sbumpc(); i = t & 0xff; t = sb.sbumpc(); i = (i << 8) | (t & 0xff); t = sb.sbumpc(); i = (i << 8) | (t & 0xff); t = sb.sbumpc(); i = (i << 8) | (t & 0xff); *ip = i; return t != std::streambuf::traits_type::eof(); } // always reads big-endian static bool deserialize_untagged_u64(uint64_t *ip, std::streambuf &sb) { std::streambuf::traits_type::int_type t; uint64_t i; t = sb.sbumpc(); i = t & 0xff; t = sb.sbumpc(); i = (i << 8) | (t & 0xff); t = sb.sbumpc(); i = (i << 8) | (t & 0xff); t = sb.sbumpc(); i = (i << 8) | (t & 0xff); t = sb.sbumpc(); i = (i << 8) | (t & 0xff); t = sb.sbumpc(); i = (i << 8) | (t & 0xff); t = sb.sbumpc(); i = (i << 8) | (t & 0xff); t = sb.sbumpc(); i = (i << 8) | (t & 0xff); *ip = i; return t != std::streambuf::traits_type::eof(); } static bool deserialize_untagged_f64(double *ip, std::streambuf &sb) { std::streambuf::traits_type::int_type t; typedef union { double id; uint64_t ii; } iu_t; iu_t iu; t = sb.sbumpc(); iu.ii = t & 0xff; t = sb.sbumpc(); iu.ii = (iu.ii<<8) | (t & 0xff); t = sb.sbumpc(); iu.ii = (iu.ii<<8) | (t & 0xff); t = sb.sbumpc(); iu.ii = (iu.ii<<8) | (t & 0xff); t = sb.sbumpc(); iu.ii = (iu.ii<<8) | (t & 0xff); t = sb.sbumpc(); iu.ii = (iu.ii<<8) | (t & 0xff); t = sb.sbumpc(); iu.ii = (iu.ii<<8) | (t & 0xff); t = sb.sbumpc(); iu.ii = (iu.ii<<8) | (t & 0xff); *ip = iu.id; return t != std::streambuf::traits_type::eof(); } static bool deserialize_tuple(pmt_t *tuple, std::streambuf &sb) { uint32_t nitems; bool ok = deserialize_untagged_u32(&nitems, sb); pmt_t list(PMT_NIL); for(uint32_t i=0; i 4){ if (i < -2147483647 || i > 2147483647) throw pmt_notimplemented("pmt_serialize (64-bit integers)", obj); } ok = serialize_untagged_u8(PST_INT32, sb); ok &= serialize_untagged_u32(i, sb); return ok; } if (pmt_is_real(obj)){ float i = pmt_to_double(obj); ok = serialize_untagged_u8(PST_DOUBLE, sb); ok &= serialize_untagged_f64(i, sb); return ok; } if (pmt_is_complex(obj)){ std::complex i = pmt_to_complex(obj); ok = serialize_untagged_u8(PST_COMPLEX, sb); ok &= serialize_untagged_f64(i.real(), sb); ok &= serialize_untagged_f64(i.imag(), sb); return ok; } } if (pmt_is_vector(obj)) { size_t vec_len = pmt::pmt_length(obj); ok = serialize_untagged_u8(PST_VECTOR, sb); ok &= serialize_untagged_u32(vec_len, sb); for(size_t i=0; i c = pmt_c32vector_ref(obj, i); ok &= serialize_untagged_f64(c.real(), sb); ok &= serialize_untagged_f64(c.imag(), sb); } return ok; } if(pmt_is_c64vector(obj)) { ok = serialize_untagged_u8(PST_UNIFORM_VECTOR, sb); ok &= serialize_untagged_u8(UVI_C64, sb); ok &= serialize_untagged_u32(vec_len, sb); ok &= serialize_untagged_u8(npad, sb); for(size_t i=0; i c = pmt_c64vector_ref(obj, i); ok &= serialize_untagged_f64(c.real(), sb); ok &= serialize_untagged_f64(c.imag(), sb); } return ok; } } if (pmt_is_dict(obj)) throw pmt_notimplemented("pmt_serialize (dict)", obj); if (pmt_is_tuple(obj)){ size_t tuple_len = pmt::pmt_length(obj); ok = serialize_untagged_u8(PST_TUPLE, sb); ok &= serialize_untagged_u32(tuple_len, sb); for(size_t i=0; i sizeof(tmpbuf)) throw pmt_notimplemented("pmt_deserialize: very long symbol", PMT_F); if (sb.sgetn(tmpbuf, u16) != u16) goto error; return pmt_intern(std::string(tmpbuf, u16)); case PST_INT32: if (!deserialize_untagged_u32(&u32, sb)) goto error; return pmt_from_long((int32_t) u32); case PST_UINT64: if(!deserialize_untagged_u64(&u64, sb)) goto error; return pmt_from_uint64(u64); case PST_PAIR: return parse_pair(sb); case PST_DOUBLE: if(!deserialize_untagged_f64(&f64, sb)) goto error; return pmt_from_double( f64 ); case PST_COMPLEX: { double r,i; if(!deserialize_untagged_f64(&r, sb) && !deserialize_untagged_f64(&i, sb)) goto error; return pmt_make_rectangular( r,i ); } case PST_TUPLE: { pmt_t tuple; if(!deserialize_tuple(&tuple, sb)){ goto error; } return tuple; } case PST_VECTOR: { uint32_t nitems; if(!deserialize_untagged_u32(&nitems, sb)) goto error; pmt_t vec = pmt_make_vector(nitems, PMT_NIL); for(uint32_t i=0; i(f64)); } return vec; } case(UVI_F64): { pmt_t vec = pmt_make_f64vector(nitems, 0); for(uint32_t i=0; i c; deserialize_untagged_f64(&f64, sb); c.real(static_cast(f64)); deserialize_untagged_f64(&f64, sb); c.imag(static_cast(f64)); pmt_c32vector_set(vec, i, c); } return vec; } case(UVI_C64): { pmt_t vec = pmt_make_c64vector(nitems, 0); for(uint32_t i=0; i c; deserialize_untagged_f64(&f64, sb); c.real(f64); deserialize_untagged_f64(&f64, sb); c.imag(f64); pmt_c64vector_set(vec, i, c); } return vec; } default: throw pmt_exception("pmt_deserialize: malformed input stream, tag value = ", pmt_from_long(tag)); } } case PST_DICT: case PST_COMMENT: throw pmt_notimplemented("pmt_deserialize: tag value = ", pmt_from_long(tag)); default: throw pmt_exception("pmt_deserialize: malformed input stream, tag value = ", pmt_from_long(tag)); } error: throw pmt_exception("pmt_deserialize: malformed input stream", PMT_F); } /* * provide a simple string accessor to the serialized pmt form */ std::string pmt_serialize_str(pmt_t obj){ std::stringbuf sb; pmt_serialize(obj, sb); return sb.str(); } /* * provide a simple string accessor to the deserialized pmt form */ pmt_t pmt_deserialize_str(std::string s){ std::stringbuf sb(s); return pmt_deserialize(sb); } /* * This is a mostly non-recursive implementation that allows us to * deserialize very long lists w/o exhausting the evaluation stack. * * On entry we've already eaten the PST_PAIR tag. */ pmt_t parse_pair(std::streambuf &sb) { uint8_t tag; pmt_t val, expr, lastnptr, nptr; // // Keep appending nodes until we get a non-PAIR cdr. // lastnptr = PMT_NIL; while (1){ expr = pmt_deserialize(sb); // read the car nptr = pmt_cons(expr, PMT_NIL); // build new cell if (pmt_is_null(lastnptr)) val = nptr; else pmt_set_cdr(lastnptr, nptr); lastnptr = nptr; if (!deserialize_untagged_u8(&tag, sb)) // get tag of cdr throw pmt_exception("pmt_deserialize: malformed input stream", PMT_F); if (tag == PST_PAIR) continue; // keep on looping... if (tag == PST_NULL){ expr = PMT_NIL; break; } // // default: push tag back and use pmt_deserialize to get the cdr // sb.sungetc(); expr = pmt_deserialize(sb); break; } // // At this point, expr contains the value of the final cdr in the list. // pmt_set_cdr(lastnptr, expr); return val; } } /* namespace pmt */