mirror of
https://github.com/WinampDesktop/winamp.git
synced 2025-01-15 17:35:17 +00:00
174 lines
5.9 KiB
C
174 lines
5.9 KiB
C
|
#ifndef __C_DATAPUMP_H__
|
||
|
#define __C_DATAPUMP_H__
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <memory.h>
|
||
|
#include <stddef.h>
|
||
|
#pragma intrinsic(memcpy,memset)
|
||
|
|
||
|
template<class T> class C_DATAPUMP {
|
||
|
private:
|
||
|
protected:
|
||
|
T *BufferBottom; // bottom of the physical buffer
|
||
|
T *BufferTop; // top of the physical buffer
|
||
|
T *BufferStart; // start of the logical buffer
|
||
|
T *BufferEnd; // end of the logical buffer
|
||
|
|
||
|
virtual void addItems(T *inputBuffer, size_t inputSize) { // inputSize = number of <T> records inputBuffer contains
|
||
|
if(inputBuffer && inputSize) {
|
||
|
memcpy(BufferEnd,inputBuffer,inputSize*sizeof(T)); // copy our records in
|
||
|
BufferEnd += inputSize;
|
||
|
if(BufferEnd >= BufferTop) BufferEnd = BufferBottom + (BufferEnd-BufferTop);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
virtual void delItems(int where, size_t numItems) { // where: 0 = start, 1 = end
|
||
|
if(numItems > 0) {
|
||
|
if(numItems > size()) { // just void everything
|
||
|
BufferEnd = BufferStart;
|
||
|
} else {
|
||
|
if(where == 0) { // start
|
||
|
BufferStart += numItems;
|
||
|
if(BufferStart >= BufferTop) BufferStart = BufferBottom + (BufferTop-BufferStart);
|
||
|
} else if(where == 1) { // end
|
||
|
BufferEnd -= numItems;
|
||
|
if(BufferEnd < BufferBottom) BufferEnd = BufferTop - (BufferBottom-BufferEnd);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
virtual void getItems(T *outputBuffer, size_t outputSize) { // outputSize = number of <T> records outputBuffer needs
|
||
|
if(outputBuffer && outputSize) {
|
||
|
memcpy(outputBuffer,BufferStart,outputSize*sizeof(T));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
C_DATAPUMP(int bufferSize) { // bufferSize = number of <T> records
|
||
|
BufferBottom = NULL;
|
||
|
BufferTop = NULL;
|
||
|
BufferStart = NULL;
|
||
|
BufferEnd = NULL;
|
||
|
resizeBuffer(bufferSize);
|
||
|
}
|
||
|
|
||
|
virtual ~C_DATAPUMP() {
|
||
|
if(getBufferSize() && BufferBottom) {
|
||
|
free(BufferBottom);
|
||
|
BufferBottom = NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
virtual void resizeBuffer(size_t bufferSize) { // bufferSize = number of <T> records
|
||
|
// this will invalidate any data in the buffer, so be careful when calling this function
|
||
|
if(bufferSize) {
|
||
|
if(getBufferSize() != bufferSize) {
|
||
|
if(BufferBottom && BufferTop && getBufferSize()) { // buffer is valid
|
||
|
if(getBufferSize() > bufferSize) { // buffer is getting smaller (will invalidate buffer)
|
||
|
BufferTop -= getBufferSize()-bufferSize;
|
||
|
invalidate();
|
||
|
} else { // buffer is getting larger (will _NOT_ invalidate buffer... nicely moves the data over =)
|
||
|
T *newBuffer = (T *)malloc(bufferSize * sizeof(T));
|
||
|
// new
|
||
|
BufferEnd = newBuffer + get(newBuffer,bufferSize);
|
||
|
free(BufferBottom);
|
||
|
BufferBottom = newBuffer;
|
||
|
BufferTop = BufferBottom + bufferSize;
|
||
|
BufferStart = BufferBottom;
|
||
|
/* old
|
||
|
T *bufptr = newBuffer;
|
||
|
int top = BufferEnd >= BufferStart ? BufferEnd-BufferStart : BufferTop-BufferStart; // number of <T> records at top of physical buffer
|
||
|
int bottom = BufferEnd >= BufferStart ? 0 : BufferEnd-BufferBottom; // number of <T> records at bottom of physical buffer
|
||
|
if(top > 0) {
|
||
|
memcpy(bufptr,BufferStart,top*sizeof(T));
|
||
|
bufptr += top;
|
||
|
}
|
||
|
if(bottom > 0) {
|
||
|
memcpy(bufptr,BufferBottom,bottom*sizeof(T));
|
||
|
bufptr += bottom;
|
||
|
}
|
||
|
free(BufferBottom);
|
||
|
BufferBottom = newBuffer;
|
||
|
BufferTop = BufferBottom + bufferSize;
|
||
|
BufferStart = BufferBottom;
|
||
|
BufferEnd = bufptr;
|
||
|
*/
|
||
|
}
|
||
|
} else { // no buffer, create (invalidates the buffer... duh)
|
||
|
BufferBottom = (T *)malloc(bufferSize * sizeof(T));
|
||
|
BufferTop = BufferBottom + bufferSize;
|
||
|
invalidate();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
virtual size_t size() { // will get the number of <T> records the logical buffer contains
|
||
|
return BufferEnd >= BufferStart ? BufferEnd-BufferStart : (BufferTop-BufferStart)+(BufferEnd-BufferBottom);
|
||
|
}
|
||
|
|
||
|
virtual size_t put(T *inputBuffer, size_t inputSize) { // inputSize = number of <T> records inputBuffer contains
|
||
|
// returns number of <T> records added to logical buffer
|
||
|
size_t retval = 0;
|
||
|
if(inputBuffer && inputSize) {
|
||
|
size_t fitting = ((BufferTop-BufferBottom)-1) - size(); // can't go over our logical boundary.... blah
|
||
|
if(fitting > inputSize) fitting = inputSize; // the entire thing can fit. yeay!
|
||
|
retval = fitting;
|
||
|
if(fitting > 0) {
|
||
|
T *bufptr = inputBuffer;
|
||
|
size_t top = BufferEnd >= BufferStart ? BufferTop-BufferEnd : 0; // number of <T> records free at top of physical buffer
|
||
|
size_t bottom = BufferEnd >= BufferStart ? BufferStart-BufferBottom : (BufferStart-BufferEnd); // number of <T> records free at bottom of physical buffer
|
||
|
if(top > 0) {
|
||
|
if(top > fitting) top = fitting;
|
||
|
addItems(bufptr,top);
|
||
|
fitting -= top;
|
||
|
bufptr += top;
|
||
|
}
|
||
|
if(bottom > 0 && fitting > 0) {
|
||
|
if(bottom > fitting) bottom = fitting;
|
||
|
addItems(bufptr,bottom);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
virtual size_t get(T *outputBuffer, size_t outputSize) { // outputSize = number of <T> records outputBuffer needs
|
||
|
// returns number of <T> records pulled from the logical buffer
|
||
|
size_t retval = 0;
|
||
|
if(outputBuffer && outputSize) {
|
||
|
size_t fitting = size();
|
||
|
if(fitting > outputSize) fitting = outputSize;
|
||
|
retval = fitting;
|
||
|
if(fitting > 0) {
|
||
|
T *bufptr = outputBuffer;
|
||
|
size_t top = BufferEnd >= BufferStart ? BufferEnd-BufferStart : BufferTop-BufferStart; // number of <T> records at top of physical buffer
|
||
|
size_t bottom = BufferEnd >= BufferStart ? 0 : BufferEnd-BufferBottom; // number of <T> records at bottom of physical buffer
|
||
|
if(top > 0) {
|
||
|
if(top > fitting) top = fitting;
|
||
|
getItems(bufptr,top);
|
||
|
delItems(0,top);
|
||
|
fitting -= top;
|
||
|
bufptr += top;
|
||
|
}
|
||
|
if(bottom > 0 && fitting > 0) {
|
||
|
if(bottom > fitting) bottom = fitting;
|
||
|
getItems(bufptr,bottom);
|
||
|
delItems(0,bottom);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
virtual size_t getBufferSize() { // returns the size of the physical buffer in <T> items
|
||
|
return BufferTop-BufferBottom;
|
||
|
}
|
||
|
|
||
|
virtual void invalidate() { // calling this will wipe all data in the buffer and reset the logical pointers
|
||
|
BufferStart = BufferEnd = BufferBottom;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
#endif // !__C_DATAPUMP_H__
|