1 #include "qemu/osdep.h" 2 #include "ui/clipboard.h" 3 4 static NotifierList clipboard_notifiers = 5 NOTIFIER_LIST_INITIALIZER(clipboard_notifiers); 6 7 static QemuClipboardInfo *cbinfo[QEMU_CLIPBOARD_SELECTION__COUNT]; 8 9 void qemu_clipboard_peer_register(QemuClipboardPeer *peer) 10 { 11 notifier_list_add(&clipboard_notifiers, &peer->notifier); 12 } 13 14 void qemu_clipboard_peer_unregister(QemuClipboardPeer *peer) 15 { 16 int i; 17 18 for (i = 0; i < QEMU_CLIPBOARD_SELECTION__COUNT; i++) { 19 qemu_clipboard_peer_release(peer, i); 20 } 21 notifier_remove(&peer->notifier); 22 } 23 24 bool qemu_clipboard_peer_owns(QemuClipboardPeer *peer, 25 QemuClipboardSelection selection) 26 { 27 QemuClipboardInfo *info = qemu_clipboard_info(selection); 28 29 return info && info->owner == peer; 30 } 31 32 void qemu_clipboard_peer_release(QemuClipboardPeer *peer, 33 QemuClipboardSelection selection) 34 { 35 g_autoptr(QemuClipboardInfo) info = NULL; 36 37 if (qemu_clipboard_peer_owns(peer, selection)) { 38 /* set empty clipboard info */ 39 info = qemu_clipboard_info_new(NULL, selection); 40 qemu_clipboard_update(info); 41 } 42 } 43 44 bool qemu_clipboard_check_serial(QemuClipboardInfo *info, bool client) 45 { 46 if (!info->has_serial || 47 !cbinfo[info->selection] || 48 !cbinfo[info->selection]->has_serial) { 49 return true; 50 } 51 52 if (client) { 53 return cbinfo[info->selection]->serial >= info->serial; 54 } else { 55 return cbinfo[info->selection]->serial > info->serial; 56 } 57 } 58 59 void qemu_clipboard_update(QemuClipboardInfo *info) 60 { 61 QemuClipboardNotify notify = { 62 .type = QEMU_CLIPBOARD_UPDATE_INFO, 63 .info = info, 64 }; 65 assert(info->selection < QEMU_CLIPBOARD_SELECTION__COUNT); 66 67 notifier_list_notify(&clipboard_notifiers, ¬ify); 68 69 qemu_clipboard_info_unref(cbinfo[info->selection]); 70 cbinfo[info->selection] = qemu_clipboard_info_ref(info); 71 } 72 73 QemuClipboardInfo *qemu_clipboard_info(QemuClipboardSelection selection) 74 { 75 assert(selection < QEMU_CLIPBOARD_SELECTION__COUNT); 76 77 return cbinfo[selection]; 78 } 79 80 QemuClipboardInfo *qemu_clipboard_info_new(QemuClipboardPeer *owner, 81 QemuClipboardSelection selection) 82 { 83 QemuClipboardInfo *info = g_new0(QemuClipboardInfo, 1); 84 85 info->owner = owner; 86 info->selection = selection; 87 info->refcount = 1; 88 89 return info; 90 } 91 92 QemuClipboardInfo *qemu_clipboard_info_ref(QemuClipboardInfo *info) 93 { 94 info->refcount++; 95 return info; 96 } 97 98 void qemu_clipboard_info_unref(QemuClipboardInfo *info) 99 { 100 uint32_t type; 101 102 if (!info) { 103 return; 104 } 105 106 info->refcount--; 107 if (info->refcount > 0) { 108 return; 109 } 110 111 for (type = 0; type < QEMU_CLIPBOARD_TYPE__COUNT; type++) { 112 g_free(info->types[type].data); 113 } 114 g_free(info); 115 } 116 117 void qemu_clipboard_request(QemuClipboardInfo *info, 118 QemuClipboardType type) 119 { 120 if (info->types[type].data || 121 info->types[type].requested || 122 !info->types[type].available || 123 !info->owner) 124 return; 125 126 info->types[type].requested = true; 127 info->owner->request(info, type); 128 } 129 130 void qemu_clipboard_reset_serial(void) 131 { 132 QemuClipboardNotify notify = { .type = QEMU_CLIPBOARD_RESET_SERIAL }; 133 134 notifier_list_notify(&clipboard_notifiers, ¬ify); 135 } 136 137 void qemu_clipboard_set_data(QemuClipboardPeer *peer, 138 QemuClipboardInfo *info, 139 QemuClipboardType type, 140 uint32_t size, 141 const void *data, 142 bool update) 143 { 144 if (!info || 145 info->owner != peer) { 146 return; 147 } 148 149 g_free(info->types[type].data); 150 info->types[type].data = g_memdup(data, size); 151 info->types[type].size = size; 152 info->types[type].available = true; 153 154 if (update) { 155 qemu_clipboard_update(info); 156 } 157 } 158