/*PVDoubleArray.cpp*/ /** * Copyright - See the COPYRIGHT that is included with this distribution. * EPICS pvDataCPP is distributed subject to a Software License Agreement found * in file LICENSE that is included with this distribution. */ #include #include #include #include #include "pvData.h" #include "factory.h" #include "serializeHelper.h" using std::min; namespace epics { namespace pvData { class BasePVDoubleArray : public PVDoubleArray { public: BasePVDoubleArray(PVStructure *parent,ScalarArrayConstPtr scalarArray); virtual ~BasePVDoubleArray(); virtual void setCapacity(int capacity); virtual int get(int offset, int length, DoubleArrayData *data) ; virtual int put(int offset,int length,DoubleArray from, int fromOffset); virtual void shareData(double value[],int capacity,int length); // from Serializable virtual void serialize(ByteBuffer *pbuffer,SerializableControl *pflusher) ; virtual void deserialize(ByteBuffer *pbuffer,DeserializableControl *pflusher); virtual void serialize(ByteBuffer *pbuffer, SerializableControl *pflusher, int offset, int count) ; virtual bool operator==(PVField& pv) ; virtual bool operator!=(PVField& pv) ; private: double *value; }; BasePVDoubleArray::BasePVDoubleArray(PVStructure *parent, ScalarArrayConstPtr scalarArray) : PVDoubleArray(parent,scalarArray),value(new double[0]) { } BasePVDoubleArray::~BasePVDoubleArray() { delete[] value; } void BasePVDoubleArray::setCapacity(int capacity) { if(getCapacity()==capacity) return; if(!isCapacityMutable()) { std::string message("not capacityMutable"); PVField::message(message, errorMessage); return; } int length = PVArray::getLength(); if(length>capacity) length = capacity; double *newValue = new double[capacity]; for(int i=0; i length) { n = length-offset; if(n<0) n = 0; } data->data = value; data->offset = offset; return n; } int BasePVDoubleArray::put(int offset,int len, DoubleArray from,int fromOffset) { if(PVField::isImmutable()) { PVField::message("field is immutable",errorMessage); return 0; } if(from==value) return len; if(len<1) return 0; int length = PVArray::getLength(); int capacity = PVArray::getCapacity(); if(offset+len > length) { int newlength = offset + len; if(newlength>capacity) { setCapacity(newlength); newlength = PVArray::getCapacity(); len = newlength - offset; if(len<=0) return 0; } length = newlength; } for(int i=0;i=0) { // prepare array, if necessary if(size>getCapacity()) setCapacity(size); // retrieve value from the buffer int i = 0; while(true) { int maxIndex = min(size-i, (int)(pbuffer->getRemaining() /sizeof(double)))+i; for(; igetDouble(); if(iensureData(sizeof(double)); else break; } // set new length setLength(size); postPut(); } // TODO null arrays (size == -1) not supported } void BasePVDoubleArray::serialize(ByteBuffer *pbuffer, SerializableControl *pflusher, int offset, int count) { // cache int length = getLength(); // check bounds if(offset<0) offset = 0; else if(offset>length) offset = length; if(count<0) count = length; int maxCount = length-offset; if(count>maxCount) count = maxCount; // write SerializeHelper::writeSize(count, pbuffer, pflusher); int end = offset+count; int i = offset; while(true) { int maxIndex = min(end-i, (int)(pbuffer->getRemaining() /sizeof(double)))+i; for(; iputDouble(value[i]); if(iflushSerializeBuffer(); else break; } } bool BasePVDoubleArray::operator==(PVField& pv) { return getConvert()->equals(this, &pv); } bool BasePVDoubleArray::operator!=(PVField& pv) { return !(getConvert()->equals(this, &pv)); } }}