aboutsummaryrefslogtreecommitdiff
path: root/Src/devices/deviceObjectEnum.cpp
diff options
context:
space:
mode:
authorJef <jef@targetspot.com>2024-09-24 08:54:57 -0400
committerJef <jef@targetspot.com>2024-09-24 08:54:57 -0400
commit20d28e80a5c861a9d5f449ea911ab75b4f37ad0d (patch)
tree12f17f78986871dd2cfb0a56e5e93b545c1ae0d0 /Src/devices/deviceObjectEnum.cpp
parent537bcbc86291b32fc04ae4133ce4d7cac8ebe9a7 (diff)
downloadwinamp-20d28e80a5c861a9d5f449ea911ab75b4f37ad0d.tar.gz
Initial community commit
Diffstat (limited to 'Src/devices/deviceObjectEnum.cpp')
-rw-r--r--Src/devices/deviceObjectEnum.cpp175
1 files changed, 175 insertions, 0 deletions
diff --git a/Src/devices/deviceObjectEnum.cpp b/Src/devices/deviceObjectEnum.cpp
new file mode 100644
index 00000000..e60b39d6
--- /dev/null
+++ b/Src/devices/deviceObjectEnum.cpp
@@ -0,0 +1,175 @@
+#include "main.h"
+#include "./deviceObjectEnum.h"
+#include <new>
+
+DeviceObjectEnum::DeviceObjectEnum()
+ : ref(1), buffer(NULL), size(0), cursor(0)
+{
+}
+
+DeviceObjectEnum::~DeviceObjectEnum()
+{
+ if (NULL != buffer)
+ {
+ while(size--)
+ {
+ buffer[size]->Release();
+ }
+ }
+}
+
+HRESULT DeviceObjectEnum::CreateInstance(ifc_deviceobject **objects,
+ size_t count,
+ DeviceObjectEnum **instance)
+{
+ size_t index, size;
+ void *storage;
+ ifc_deviceobject *o;
+ DeviceObjectEnum *enumerator;
+
+
+ if (NULL == instance)
+ return E_POINTER;
+
+ *instance = NULL;
+
+ size = sizeof(DeviceObjectEnum) + (sizeof(ifc_deviceobject**) * count);
+ storage = calloc(1, size);
+ if (NULL == storage)
+ return E_OUTOFMEMORY;
+
+ enumerator = new(storage) DeviceObjectEnum();
+ if (NULL == enumerator)
+ {
+ free(storage);
+ return E_FAIL;
+ }
+
+ enumerator->buffer = (ifc_deviceobject**)(((BYTE*)enumerator) + sizeof(DeviceObjectEnum));
+
+ for (index = 0; index < count; index++)
+ {
+ o = objects[index];
+ if (NULL != o)
+ {
+ enumerator->buffer[enumerator->size] = o;
+ o->AddRef();
+ enumerator->size++;
+ }
+ }
+
+ *instance = enumerator;
+ return S_OK;
+}
+
+
+
+size_t DeviceObjectEnum::AddRef()
+{
+ return InterlockedIncrement((LONG*)&ref);
+}
+
+size_t DeviceObjectEnum::Release()
+{
+ if (0 == ref)
+ return ref;
+
+ LONG r = InterlockedDecrement((LONG*)&ref);
+ if (0 == r)
+ delete(this);
+
+ return r;
+}
+
+int DeviceObjectEnum::QueryInterface(GUID interface_guid, void **object)
+{
+ if (NULL == object) return E_POINTER;
+
+ if (IsEqualIID(interface_guid, IFC_DeviceObjectEnum))
+ *object = static_cast<ifc_deviceobjectenum*>(this);
+ else
+ {
+ *object = NULL;
+ return E_NOINTERFACE;
+ }
+
+ if (NULL == *object)
+ return E_UNEXPECTED;
+
+ AddRef();
+ return S_OK;
+}
+
+HRESULT DeviceObjectEnum::Next(ifc_deviceobject **objects, size_t bufferMax, size_t *fetched)
+{
+ size_t available, copied, index;
+ ifc_deviceobject **source;
+
+ if (NULL == objects)
+ return E_POINTER;
+
+ if (0 == bufferMax)
+ return E_INVALIDARG;
+
+ if (cursor >= size)
+ {
+ if (NULL != fetched)
+ *fetched = 0;
+
+ return S_FALSE;
+ }
+
+ available = size - cursor;
+ copied = ((available > bufferMax) ? bufferMax : available);
+
+ source = buffer + cursor;
+ CopyMemory(objects, source, copied * sizeof(ifc_deviceobject*));
+
+ for(index = 0; index < copied; index++)
+ objects[index]->AddRef();
+
+ cursor += copied;
+
+ if (NULL != fetched)
+ *fetched = copied;
+
+ return (bufferMax == copied) ? S_OK : S_FALSE;
+}
+
+HRESULT DeviceObjectEnum::Reset(void)
+{
+ cursor = 0;
+ return S_OK;
+}
+
+HRESULT DeviceObjectEnum::Skip(size_t count)
+{
+ cursor += count;
+ if (cursor > size)
+ cursor = size;
+
+ return (cursor < size) ? S_OK : S_FALSE;
+}
+
+
+HRESULT DeviceObjectEnum::GetCount(size_t *count)
+{
+ if (NULL == count)
+ return E_POINTER;
+
+ *count = size;
+
+ return S_OK;
+}
+
+#define CBCLASS DeviceObjectEnum
+START_DISPATCH;
+CB(ADDREF, AddRef)
+CB(RELEASE, Release)
+CB(QUERYINTERFACE, QueryInterface)
+CB(API_NEXT, Next)
+CB(API_RESET, Reset)
+CB(API_SKIP, Skip)
+CB(API_GETCOUNT, GetCount)
+END_DISPATCH;
+#undef CBCLASS \ No newline at end of file