CSUSBManager.m 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. //
  2. // Copyright © 2021 osy. All rights reserved.
  3. //
  4. // Licensed under the Apache License, Version 2.0 (the "License");
  5. // you may not use this file except in compliance with the License.
  6. // You may obtain a copy of the License at
  7. //
  8. // http://www.apache.org/licenses/LICENSE-2.0
  9. //
  10. // Unless required by applicable law or agreed to in writing, software
  11. // distributed under the License is distributed on an "AS IS" BASIS,
  12. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. // See the License for the specific language governing permissions and
  14. // limitations under the License.
  15. //
  16. #import "CocoaSpice.h"
  17. #import <glib.h>
  18. #import <spice-client.h>
  19. typedef enum {
  20. kUsbManagerCallConnect,
  21. kUsbManagerCallDisconnect
  22. } usbManagerCall;
  23. typedef struct {
  24. usbManagerCall call;
  25. SpiceUsbDeviceManager *manager;
  26. SpiceUsbDevice *device;
  27. gpointer callback;
  28. } usbManagerData;
  29. @interface CSUSBManager ()
  30. @property (nonatomic, readwrite, nonnull) SpiceUsbDeviceManager *usbDeviceManager;
  31. @end
  32. @implementation CSUSBManager
  33. #pragma mark - Signal callbacks
  34. static void cs_device_error(SpiceUsbDeviceManager *manager,
  35. SpiceUsbDevice *device,
  36. GError *error,
  37. gpointer data)
  38. {
  39. CSUSBManager *self = (__bridge CSUSBManager *)data;
  40. CSUSBDevice *usbdevice = [CSUSBDevice usbDeviceWithDevice:device];
  41. if (error->domain == G_IO_ERROR && error->code == G_IO_ERROR_CANCELLED)
  42. return;
  43. [self.delegate spiceUsbManager:self deviceError:[NSString stringWithUTF8String:error->message] forDevice:usbdevice];
  44. }
  45. static void cs_device_added(SpiceUsbDeviceManager *manager,
  46. SpiceUsbDevice *device, gpointer data)
  47. {
  48. CSUSBManager *self = (__bridge CSUSBManager *)data;
  49. CSUSBDevice *usbdevice = [CSUSBDevice usbDeviceWithDevice:device];
  50. [self.delegate spiceUsbManager:self deviceAttached:usbdevice];
  51. }
  52. static void cs_device_removed(SpiceUsbDeviceManager *manager,
  53. SpiceUsbDevice *device, gpointer data)
  54. {
  55. CSUSBManager *self = (__bridge CSUSBManager *)data;
  56. CSUSBDevice *usbdevice = [CSUSBDevice usbDeviceWithDevice:device];
  57. [self.delegate spiceUsbManager:self deviceRemoved:usbdevice];
  58. }
  59. static void cs_connect_cb(GObject *gobject, GAsyncResult *res, gpointer data)
  60. {
  61. SpiceUsbDeviceManager *manager = SPICE_USB_DEVICE_MANAGER(gobject);
  62. CSUSBManagerConnectionCallback callback = (__bridge_transfer CSUSBManagerConnectionCallback)(data);
  63. GError *err = NULL;
  64. spice_usb_device_manager_connect_device_finish(manager, res, &err);
  65. if (err) {
  66. callback(NO, [NSString stringWithUTF8String:err->message]);
  67. g_error_free(err);
  68. } else {
  69. callback(YES, nil);
  70. }
  71. }
  72. static void cs_disconnect_cb(GObject *gobject, GAsyncResult *res, gpointer data)
  73. {
  74. SpiceUsbDeviceManager *manager = SPICE_USB_DEVICE_MANAGER(gobject);
  75. CSUSBManagerConnectionCallback callback = (__bridge_transfer CSUSBManagerConnectionCallback)(data);
  76. GError *err = NULL;
  77. spice_usb_device_manager_disconnect_device_finish(manager, res, &err);
  78. if (err) {
  79. callback(NO, [NSString stringWithUTF8String:err->message]);
  80. g_error_free(err);
  81. } else {
  82. callback(YES, nil);
  83. }
  84. }
  85. static gboolean cs_call_manager(gpointer user_data)
  86. {
  87. usbManagerData *data = (usbManagerData *)user_data;
  88. switch (data->call) {
  89. case kUsbManagerCallConnect:
  90. spice_usb_device_manager_connect_device_async(data->manager, data->device, NULL, cs_connect_cb, data->callback);
  91. break;
  92. case kUsbManagerCallDisconnect:
  93. spice_usb_device_manager_disconnect_device_async(data->manager, data->device, NULL, cs_disconnect_cb, data->callback);
  94. break;
  95. default:
  96. g_assert(0);
  97. }
  98. return G_SOURCE_REMOVE;
  99. }
  100. #pragma mark - Properties
  101. - (BOOL)isAutoConnect {
  102. gboolean value;
  103. g_object_get(self.usbDeviceManager, "auto-connect", &value, NULL);
  104. return value;
  105. }
  106. - (void)setIsAutoConnect:(BOOL)isAutoConnect {
  107. g_object_set(self.usbDeviceManager, "auto-connect", isAutoConnect, NULL);
  108. }
  109. - (NSString *)autoConnectFilter {
  110. gchar *string;
  111. g_object_get(self.usbDeviceManager, "auto-connect-filter", &string, NULL);
  112. NSString *nsstring = [NSString stringWithUTF8String:string];
  113. g_free(string);
  114. return nsstring;
  115. }
  116. - (void)setAutoConnectFilter:(NSString *)autoConnectFilter {
  117. const gchar *string = [autoConnectFilter UTF8String];
  118. g_object_set(self.usbDeviceManager, "auto-connect-filter", string, NULL);
  119. }
  120. - (BOOL)isRedirectOnConnect {
  121. gboolean value;
  122. g_object_get(self.usbDeviceManager, "redirect-on-connect", &value, NULL);
  123. return value;
  124. }
  125. - (void)setIsRedirectOnConnect:(BOOL)isRedirectOnConnect {
  126. g_object_set(self.usbDeviceManager, "redirect-on-connect", isRedirectOnConnect, NULL);
  127. }
  128. - (NSInteger)numberFreeChannels {
  129. gint value;
  130. g_object_get(self.usbDeviceManager, "free-channels", &value, NULL);
  131. return value;
  132. }
  133. - (NSArray<CSUSBDevice *> *)usbDevices {
  134. NSMutableArray<CSUSBDevice *> *usbDevices = [NSMutableArray new];
  135. GPtrArray *arr = spice_usb_device_manager_get_devices(self.usbDeviceManager);
  136. if (arr != NULL) {
  137. for (int i = 0; i < arr->len; i++) {
  138. SpiceUsbDevice *device = g_ptr_array_index(arr, i);
  139. [usbDevices addObject:[CSUSBDevice usbDeviceWithDevice:device]];
  140. }
  141. g_ptr_array_unref(arr);
  142. }
  143. return usbDevices;
  144. }
  145. - (BOOL)isBusy {
  146. return spice_usb_device_manager_is_redirecting(self.usbDeviceManager);
  147. }
  148. #pragma mark - Construction
  149. - (instancetype)initWithUsbDeviceManager:(SpiceUsbDeviceManager *)usbDeviceManager {
  150. if (self = [super init]) {
  151. self.usbDeviceManager = usbDeviceManager;
  152. g_signal_connect(usbDeviceManager, "auto-connect-failed",
  153. G_CALLBACK(cs_device_error), GLIB_OBJC_RETAIN(self));
  154. g_signal_connect(usbDeviceManager, "device-error",
  155. G_CALLBACK(cs_device_error), GLIB_OBJC_RETAIN(self));
  156. g_signal_connect(usbDeviceManager, "device-added",
  157. G_CALLBACK(cs_device_added), GLIB_OBJC_RETAIN(self));
  158. g_signal_connect(usbDeviceManager, "device-removed",
  159. G_CALLBACK(cs_device_removed), GLIB_OBJC_RETAIN(self));
  160. }
  161. return self;
  162. }
  163. - (void)dealloc {
  164. g_signal_handlers_disconnect_by_func(self.usbDeviceManager, G_CALLBACK(cs_device_error), GLIB_OBJC_RELEASE(self));
  165. g_signal_handlers_disconnect_by_func(self.usbDeviceManager, G_CALLBACK(cs_device_error), GLIB_OBJC_RELEASE(self));
  166. g_signal_handlers_disconnect_by_func(self.usbDeviceManager, G_CALLBACK(cs_device_added), GLIB_OBJC_RELEASE(self));
  167. g_signal_handlers_disconnect_by_func(self.usbDeviceManager, G_CALLBACK(cs_device_removed), GLIB_OBJC_RELEASE(self));
  168. }
  169. #pragma mark - Methods
  170. - (BOOL)canRedirectUsbDevice:(CSUSBDevice *)usbDevice errorMessage:(NSString * _Nullable __autoreleasing *)errorMessage {
  171. GError *err = NULL;
  172. gboolean res = spice_usb_device_manager_can_redirect_device(self.usbDeviceManager, usbDevice.device, &err);
  173. if (errorMessage && err) {
  174. *errorMessage = [NSString stringWithUTF8String:err->message];
  175. }
  176. g_clear_error(&err);
  177. return res;
  178. }
  179. - (BOOL)isUsbDeviceConnected:(CSUSBDevice *)usbDevice {
  180. return spice_usb_device_manager_is_device_connected(self.usbDeviceManager, usbDevice.device);
  181. }
  182. - (void)spiceUsbManagerCall:(usbManagerCall)call forUsbDevice:(CSUSBDevice *)usbDevice withCompletion:(CSUSBManagerConnectionCallback)completion {
  183. usbManagerData *data = g_new0(usbManagerData, 1);
  184. data->call = call;
  185. data->manager = self.usbDeviceManager;
  186. data->device = usbDevice.device;
  187. data->callback = (__bridge_retained gpointer)completion;
  188. g_main_context_invoke_full([CSMain sharedInstance].glibMainContext,
  189. G_PRIORITY_HIGH,
  190. cs_call_manager,
  191. data,
  192. g_free);
  193. }
  194. - (void)connectUsbDevice:(CSUSBDevice *)usbDevice withCompletion:(CSUSBManagerConnectionCallback)completion {
  195. [self spiceUsbManagerCall:kUsbManagerCallConnect forUsbDevice:usbDevice withCompletion:completion];
  196. }
  197. - (void)disconnectUsbDevice:(CSUSBDevice *)usbDevice withCompletion:(CSUSBManagerConnectionCallback)completion {
  198. [self spiceUsbManagerCall:kUsbManagerCallDisconnect forUsbDevice:usbDevice withCompletion:completion];
  199. }
  200. @end