diff options
| author | Gordon Sim <gsim@apache.org> | 2008-10-24 11:33:40 +0000 |
|---|---|---|
| committer | Gordon Sim <gsim@apache.org> | 2008-10-24 11:33:40 +0000 |
| commit | 93b372e4d8d51a3db161a95a8b1884ce6af117fd (patch) | |
| tree | 4a48e33e5042ea1f3a7be46100d3d6a9c73d5c09 /cpp/src/qpid | |
| parent | 55c976dbede7ada5dbcc581945f7d5b1a038344c (diff) | |
| download | qpid-python-93b372e4d8d51a3db161a95a8b1884ce6af117fd.tar.gz | |
Fix for bug in encoding/decoding of floats and doubles.
Added test to verify double/float headers set by c++ are correctly read by python
[TODO: add wider range of header types, add bidirectional tests, add tests against other languages]
[Also fixed one or two SSL related files missing from distribution list]
git-svn-id: https://svn.apache.org/repos/asf/incubator/qpid/trunk/qpid@707604 13f79535-47bb-0310-9956-ffa450edef68
Diffstat (limited to 'cpp/src/qpid')
| -rw-r--r-- | cpp/src/qpid/framing/Endian.cpp | 52 | ||||
| -rw-r--r-- | cpp/src/qpid/framing/Endian.h | 46 | ||||
| -rw-r--r-- | cpp/src/qpid/framing/FieldTable.cpp | 5 | ||||
| -rw-r--r-- | cpp/src/qpid/framing/FieldValue.cpp | 5 | ||||
| -rw-r--r-- | cpp/src/qpid/framing/FieldValue.h | 2 |
5 files changed, 106 insertions, 4 deletions
diff --git a/cpp/src/qpid/framing/Endian.cpp b/cpp/src/qpid/framing/Endian.cpp new file mode 100644 index 0000000000..a2d4566e5e --- /dev/null +++ b/cpp/src/qpid/framing/Endian.cpp @@ -0,0 +1,52 @@ +/* + * + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * + */ +#include "Endian.h" + +namespace qpid { +namespace framing { + +Endian::Endian() : littleEndian(!testBigEndian()) {} + +bool Endian::testBigEndian() +{ + uint16_t a = 1; + uint16_t b; + uint8_t* p = (uint8_t*) &b; + p[0] = 0xFF & (a >> 8); + p[1] = 0xFF & (a); + return a == b; +} + +uint8_t* const Endian::convertIfRequired(uint8_t* const octets, int width) +{ + if (instance.littleEndian) { + for (int i = 0; i < (width/2); i++) { + uint8_t temp = octets[i]; + octets[i] = octets[width - (1 + i)]; + octets[width - (1 + i)] = temp; + } + } + return octets; +} + +const Endian Endian::instance; + +}} // namespace qpid::framing diff --git a/cpp/src/qpid/framing/Endian.h b/cpp/src/qpid/framing/Endian.h new file mode 100644 index 0000000000..14d3f9e66d --- /dev/null +++ b/cpp/src/qpid/framing/Endian.h @@ -0,0 +1,46 @@ +#ifndef QPID_FRAMING_ENDIAN_H +#define QPID_FRAMING_ENDIAN_H + +/* + * + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * + */ + +#include "qpid/sys/IntegerTypes.h" + +namespace qpid { +namespace framing { + +/** + * Conversion utility for little-endian platforms that need to convert + * to and from network ordered octet sequences + */ +class Endian +{ + public: + static uint8_t* const convertIfRequired(uint8_t* const octets, int width); + private: + const bool littleEndian; + Endian(); + static const Endian instance; + static bool testBigEndian(); +}; +}} // namespace qpid::framing + +#endif /*!QPID_FRAMING_ENDIAN_H*/ diff --git a/cpp/src/qpid/framing/FieldTable.cpp b/cpp/src/qpid/framing/FieldTable.cpp index cf0e03180c..90acbaf6bc 100644 --- a/cpp/src/qpid/framing/FieldTable.cpp +++ b/cpp/src/qpid/framing/FieldTable.cpp @@ -21,6 +21,7 @@ #include "FieldTable.h" #include "Array.h" #include "Buffer.h" +#include "Endian.h" #include "FieldValue.h" #include "qpid/Exception.h" #include "qpid/framing/reply_exceptions.h" @@ -157,7 +158,9 @@ bool getRawFixedWidthValue(FieldTable::ValuePtr vptr, T& value) if (vptr && vptr->getType() == typecode) { FixedWidthValue<width>* fwv = dynamic_cast< FixedWidthValue<width>* >(&vptr->getData()); if (fwv) { - fwv->copyInto(reinterpret_cast<uint8_t*>(&value)); + uint8_t* const octets = Endian::convertIfRequired(fwv->rawOctets(), width); + uint8_t* const target = reinterpret_cast<uint8_t*>(&value); + for (uint i = 0; i < width; ++i) target[i] = octets[i]; return true; } } diff --git a/cpp/src/qpid/framing/FieldValue.cpp b/cpp/src/qpid/framing/FieldValue.cpp index 5fbfe7d0c1..9107ceeeea 100644 --- a/cpp/src/qpid/framing/FieldValue.cpp +++ b/cpp/src/qpid/framing/FieldValue.cpp @@ -21,6 +21,7 @@ #include "FieldValue.h" #include "Array.h" #include "Buffer.h" +#include "Endian.h" #include "qpid/framing/reply_exceptions.h" namespace qpid { @@ -138,11 +139,11 @@ IntegerValue::IntegerValue(int v) : {} FloatValue::FloatValue(float v) : - FieldValue(0x23, new FixedWidthValue<4>(reinterpret_cast<uint8_t*>(&v))) + FieldValue(0x23, new FixedWidthValue<4>(Endian::convertIfRequired(reinterpret_cast<uint8_t*>(&v), 4))) {} DoubleValue::DoubleValue(double v) : - FieldValue(0x33, new FixedWidthValue<8>(reinterpret_cast<uint8_t*>(&v))) + FieldValue(0x33, new FixedWidthValue<8>(Endian::convertIfRequired(reinterpret_cast<uint8_t*>(&v), 8))) {} Integer64Value::Integer64Value(int64_t v) : diff --git a/cpp/src/qpid/framing/FieldValue.h b/cpp/src/qpid/framing/FieldValue.h index 0a70360cbd..18c45f3ff8 100644 --- a/cpp/src/qpid/framing/FieldValue.h +++ b/cpp/src/qpid/framing/FieldValue.h @@ -166,6 +166,7 @@ class FixedWidthValue : public FieldValue::Data { { for (uint i = 0; i < width; ++i) data[i] = octets[i]; } + uint8_t* const rawOctets() { return octets; } void print(std::ostream& o) const { o << "F" << width << ":"; }; }; @@ -304,7 +305,6 @@ class ArrayValue : public FieldValue { ArrayValue(const Array&); }; - template <class T> bool getEncodedValue(FieldTable::ValuePtr vptr, T& value) { |
