#ifndef CDEVICEFINDER_HPP #define CDEVICEFINDER_HPP #include #include #include #include "DeviceToken.hpp" #include "IHIDListener.hpp" #include "DeviceSignature.hpp" #include #include #ifdef _WIN32 #define _WIN32_LEAN_AND_MEAN 1 #include #include #endif namespace boo { static class DeviceFinder* skDevFinder = NULL; class DeviceFinder { public: friend class HIDListenerIOKit; friend class HIDListenerUdev; friend class HIDListenerWinUSB; static inline DeviceFinder* instance() {return skDevFinder;} private: /* Types this finder is interested in (immutable) */ DeviceSignature::TDeviceSignatureSet m_types; /* Platform-specific USB event registration * (for auto-scanning, NULL if not registered) */ IHIDListener* m_listener; /* Set of presently-connected device tokens */ TDeviceTokens m_tokens; std::mutex m_tokensLock; /* Friend methods for platform-listener to find/insert/remove * tokens with type-filtering */ inline bool _hasToken(const std::string& path) { auto preCheck = m_tokens.find(path); if (preCheck != m_tokens.end()) return true; return false; } inline bool _insertToken(DeviceToken&& token) { if (DeviceSignature::DeviceMatchToken(token, m_types)) { m_tokensLock.lock(); TInsertedDeviceToken inseredTok = m_tokens.insert(std::make_pair(token.getDevicePath(), std::move(token))); m_tokensLock.unlock(); deviceConnected(inseredTok.first->second); return true; } return false; } inline void _removeToken(const std::string& path) { auto preCheck = m_tokens.find(path); if (preCheck != m_tokens.end()) { DeviceToken& tok = preCheck->second; DeviceBase* dev = tok.m_connectedDev; tok._deviceClose(); deviceDisconnected(tok, dev); m_tokensLock.lock(); m_tokens.erase(preCheck); m_tokensLock.unlock(); } } public: class CDeviceTokensHandle { DeviceFinder& m_finder; public: inline CDeviceTokensHandle(DeviceFinder& finder) : m_finder(finder) {m_finder.m_tokensLock.lock();} inline ~CDeviceTokensHandle() {m_finder.m_tokensLock.unlock();} inline TDeviceTokens::iterator begin() {return m_finder.m_tokens.begin();} inline TDeviceTokens::iterator end() {return m_finder.m_tokens.end();} }; /* Application must specify its interested device-types */ DeviceFinder(std::unordered_set types) : m_listener(NULL) { if (skDevFinder) { fprintf(stderr, "only one instance of CDeviceFinder may be constructed"); abort(); } skDevFinder = this; for (const std::type_index& typeIdx : types) { const DeviceSignature* sigIter = BOO_DEVICE_SIGS; while (sigIter->m_name) { if (sigIter->m_typeIdx == typeIdx) m_types.push_back(sigIter); ++sigIter; } } } ~DeviceFinder() { if (m_listener) m_listener->stopScanning(); delete m_listener; skDevFinder = NULL; } /* Get interested device-type mask */ inline const DeviceSignature::TDeviceSignatureSet& getTypes() const {return m_types;} /* Iterable set of tokens */ inline CDeviceTokensHandle getTokens() {return CDeviceTokensHandle(*this);} /* Automatic device scanning */ inline bool startScanning() { if (!m_listener) m_listener = IHIDListenerNew(*this); if (m_listener) return m_listener->startScanning(); return false; } inline bool stopScanning() { if (!m_listener) m_listener = IHIDListenerNew(*this); if (m_listener) return m_listener->stopScanning(); return false; } /* Manual device scanning */ inline bool scanNow() { if (!m_listener) m_listener = IHIDListenerNew(*this); if (m_listener) return m_listener->scanNow(); return false; } virtual void deviceConnected(DeviceToken&) {} virtual void deviceDisconnected(DeviceToken&, DeviceBase*) {} #if _WIN32 /* Windows-specific WM_DEVICECHANGED handler */ static LRESULT winDevChangedHandler(WPARAM wParam, LPARAM lParam) { PDEV_BROADCAST_HDR dbh = (PDEV_BROADCAST_HDR)lParam; PDEV_BROADCAST_DEVICEINTERFACE dbhi = (PDEV_BROADCAST_DEVICEINTERFACE)lParam; CDeviceFinder* finder = instance(); if (!finder) return 0; if (wParam == DBT_DEVICEARRIVAL) { if (dbh->dbch_devicetype == DBT_DEVTYP_DEVICEINTERFACE) { #ifdef UNICODE char devPath[1024]; wcstombs(devPath, dbhi->dbcc_name, 1024); finder->m_listener->_extDevConnect(devPath); #else finder->m_listener->_extDevConnect(dbhi->dbcc_name); #endif } } else if (wParam == DBT_DEVICEREMOVECOMPLETE) { if (dbh->dbch_devicetype == DBT_DEVTYP_DEVICEINTERFACE) { #ifdef UNICODE char devPath[1024]; wcstombs(devPath, dbhi->dbcc_name, 1024); finder->m_listener->_extDevDisconnect(devPath); #else finder->m_listener->_extDevDisconnect(dbhi->dbcc_name); #endif } } return 0; } #endif }; } #endif // CDEVICEFINDER_HPP