/* ---------------------------------------------------------------------- * Project: CMSIS DSP Library * Title: Sched.h * Description: C++ support templates for the SDF scheduler * * $Date: 29 July 2021 * $Revision: V1.10.0 * * Target Processor: Cortex-M and Cortex-A cores * -------------------------------------------------------------------- */ /* * Copyright (C) 2010-2021 ARM Limited or its affiliates. All rights reserved. * * SPDX-License-Identifier: Apache-2.0 * * Licensed 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 * * 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. */ #ifndef _SCHEDGEN_H_ #define _SCHEDGEN_H_ #include // FIFOS #ifdef DEBUGSCHED template struct debugtype{ typedef T type; }; template<> struct debugtype{ typedef int type; }; template using Debug = struct debugtype; #endif template class FIFOBase{ public: virtual T* getWriteBuffer(int nb)=0; virtual T* getReadBuffer(int nb)=0; }; template class FIFO: public FIFOBase { public: FIFO(T *buffer,int delay=0):mBuffer(buffer),readPos(0),writePos(delay) {}; FIFO(uint8_t *buffer,int delay=0):mBuffer((T*)buffer),readPos(0),writePos(delay) {}; T * getWriteBuffer(int nb) override { if (isArray==1) { return(mBuffer); } T *ret; if (readPos > 0) { memcpy((void*)mBuffer,(void*)(mBuffer+readPos),(writePos-readPos)*sizeof(T)); writePos -= readPos; readPos = 0; } ret = mBuffer + writePos; writePos += nb; return(ret); }; T* getReadBuffer(int nb) override { if (isArray==1) { return(mBuffer); } T *ret = mBuffer + readPos; readPos += nb; return(ret); } #ifdef DEBUGSCHED void dump() { int nb=0; std::cout << std::endl; for(int i=0; i < length ; i++) { std::cout << (Debug::type)mBuffer[i] << " "; nb++; if (nb == 10) { nb=0; std::cout << std::endl; } } std::cout << std::endl; std::cout << std::endl; } #endif protected: T *mBuffer; int readPos,writePos; }; // GENERIC NODES class NodeBase { public: virtual int run()=0; }; template class GenericNode:public NodeBase { public: GenericNode(FIFOBase &src,FIFOBase &dst):mSrc(src),mDst(dst){}; protected: OUT * getWriteBuffer(){return mDst.getWriteBuffer(outputSize);}; IN * getReadBuffer(){return mSrc.getReadBuffer(inputSize);}; private: FIFOBase &mSrc; FIFOBase &mDst; }; template class GenericNode12:public NodeBase { public: GenericNode12(FIFOBase &src,FIFOBase &dst1,FIFOBase &dst2):mSrc(src), mDst1(dst1),mDst2(dst2){}; protected: OUT1 * getWriteBuffer1(){return mDst1.getWriteBuffer(output1Size);}; OUT2 * getWriteBuffer2(){return mDst2.getWriteBuffer(output2Size);}; IN * getReadBuffer(){return mSrc.getReadBuffer(inputSize);}; private: FIFOBase &mSrc; FIFOBase &mDst1; FIFOBase &mDst2; }; template class GenericNode21:public NodeBase { public: GenericNode21(FIFOBase &src1,FIFOBase &src2,FIFOBase &dst):mSrc1(src1), mSrc2(src2), mDst(dst){}; protected: OUT * getWriteBuffer(){return mDst.getWriteBuffer(outputSize);}; IN1 * getReadBuffer1(){return mSrc1.getReadBuffer(input1Size);}; IN2 * getReadBuffer2(){return mSrc2.getReadBuffer(input2Size);}; private: FIFOBase &mSrc1; FIFOBase &mSrc2; FIFOBase &mDst; }; template class GenericSource:public NodeBase { public: GenericSource(FIFOBase &dst):mDst(dst){}; protected: OUT * getWriteBuffer(){return mDst.getWriteBuffer(outputSize);}; private: FIFOBase &mDst; }; template class GenericSink:public NodeBase { public: GenericSink(FIFOBase &src):mSrc(src){}; protected: IN * getReadBuffer(){return mSrc.getReadBuffer(inputSize);}; private: FIFOBase &mSrc; }; #define REPEAT(N) for(int i=0;i class SlidingBuffer: public GenericNode { public: SlidingBuffer(FIFOBase &src,FIFOBase &dst):GenericNode(src,dst) { static_assert((windowSize-overlap)>0, "Overlap is too big"); memory.resize(overlap); }; int run(){ IN *a=this->getReadBuffer(); IN *b=this->getWriteBuffer(); memcpy((void*)b,(void*)memory.data(),overlap*sizeof(IN)); memcpy((void*)(b+overlap),(void*)a,(windowSize-overlap)*sizeof(IN)); memcpy((void*)memory.data(),(void*)(b+windowSize-overlap),overlap*sizeof(IN)) ; return(0); }; protected: std::vector memory; }; template class OverlapAdd: public GenericNode { public: OverlapAdd(FIFOBase &src,FIFOBase &dst):GenericNode(src,dst) { static_assert((windowSize-overlap)>0, "Overlap is too big"); memory.resize(overlap); }; int run(){ int i; IN *a=this->getReadBuffer(); IN *b=this->getWriteBuffer(); for(i=0;i 0) { memcpy((void*)b,(void*)memory.data(),(windowSize-overlap)*sizeof(IN)); memmove(memory.data(),memory.data()+windowSize-overlap,(2*overlap - windowSize)*sizeof(IN)); memcpy(memory.data()+2*overlap - windowSize,a+overlap,(windowSize-overlap)*sizeof(IN)); } else if (2*overlap - windowSize < 0) { memcpy((void*)b,(void*)memory.data(),overlap*sizeof(IN)); memcpy((void*)(b+overlap),(void*)(a+overlap),(windowSize - 2*overlap)*sizeof(IN)); memcpy((void*)memory.data(),(void*)(a+windowSize-overlap),overlap*sizeof(IN)); } else { memcpy((void*)b,(void*)memory.data(),overlap*sizeof(IN)); memcpy((void*)memory.data(),(void*)(a+overlap),overlap*sizeof(IN)); } return(0); }; protected: std::vector memory; }; #if !defined(CHECKERROR) #define CHECKERROR if (sdfError < 0) \ {\ break;\ } #endif #endif