1660e8d0fSGerd Hoffmann #include "qemu/osdep.h" 2660e8d0fSGerd Hoffmann #include "ui/clipboard.h" 3660e8d0fSGerd Hoffmann 4660e8d0fSGerd Hoffmann static NotifierList clipboard_notifiers = 5660e8d0fSGerd Hoffmann NOTIFIER_LIST_INITIALIZER(clipboard_notifiers); 6660e8d0fSGerd Hoffmann 7684e64d3SMarc-André Lureau static QemuClipboardInfo *cbinfo[QEMU_CLIPBOARD_SELECTION__COUNT]; 8684e64d3SMarc-André Lureau 9660e8d0fSGerd Hoffmann void qemu_clipboard_peer_register(QemuClipboardPeer *peer) 10660e8d0fSGerd Hoffmann { 11*1b17f1e9SMarc-André Lureau notifier_list_add(&clipboard_notifiers, &peer->notifier); 12660e8d0fSGerd Hoffmann } 13660e8d0fSGerd Hoffmann 14660e8d0fSGerd Hoffmann void qemu_clipboard_peer_unregister(QemuClipboardPeer *peer) 15660e8d0fSGerd Hoffmann { 167424bfaaSMarc-André Lureau int i; 177424bfaaSMarc-André Lureau 187424bfaaSMarc-André Lureau for (i = 0; i < QEMU_CLIPBOARD_SELECTION__COUNT; i++) { 197424bfaaSMarc-André Lureau qemu_clipboard_peer_release(peer, i); 207424bfaaSMarc-André Lureau } 21*1b17f1e9SMarc-André Lureau notifier_remove(&peer->notifier); 22660e8d0fSGerd Hoffmann } 23660e8d0fSGerd Hoffmann 24482bbaf4SMarc-André Lureau bool qemu_clipboard_peer_owns(QemuClipboardPeer *peer, 25482bbaf4SMarc-André Lureau QemuClipboardSelection selection) 26482bbaf4SMarc-André Lureau { 27482bbaf4SMarc-André Lureau QemuClipboardInfo *info = qemu_clipboard_info(selection); 28482bbaf4SMarc-André Lureau 29482bbaf4SMarc-André Lureau return info && info->owner == peer; 30482bbaf4SMarc-André Lureau } 31482bbaf4SMarc-André Lureau 321387865eSMarc-André Lureau void qemu_clipboard_peer_release(QemuClipboardPeer *peer, 331387865eSMarc-André Lureau QemuClipboardSelection selection) 341387865eSMarc-André Lureau { 351387865eSMarc-André Lureau g_autoptr(QemuClipboardInfo) info = NULL; 361387865eSMarc-André Lureau 371387865eSMarc-André Lureau if (qemu_clipboard_peer_owns(peer, selection)) { 381387865eSMarc-André Lureau /* set empty clipboard info */ 391387865eSMarc-André Lureau info = qemu_clipboard_info_new(NULL, selection); 401387865eSMarc-André Lureau qemu_clipboard_update(info); 411387865eSMarc-André Lureau } 421387865eSMarc-André Lureau } 431387865eSMarc-André Lureau 44660e8d0fSGerd Hoffmann void qemu_clipboard_update(QemuClipboardInfo *info) 45660e8d0fSGerd Hoffmann { 46*1b17f1e9SMarc-André Lureau QemuClipboardNotify notify = { 47*1b17f1e9SMarc-André Lureau .type = QEMU_CLIPBOARD_UPDATE_INFO, 48*1b17f1e9SMarc-André Lureau .info = info, 49*1b17f1e9SMarc-André Lureau }; 50684e64d3SMarc-André Lureau g_autoptr(QemuClipboardInfo) old = NULL; 51*1b17f1e9SMarc-André Lureau 52684e64d3SMarc-André Lureau assert(info->selection < QEMU_CLIPBOARD_SELECTION__COUNT); 53684e64d3SMarc-André Lureau 54*1b17f1e9SMarc-André Lureau notifier_list_notify(&clipboard_notifiers, ¬ify); 55684e64d3SMarc-André Lureau 56684e64d3SMarc-André Lureau old = cbinfo[info->selection]; 57684e64d3SMarc-André Lureau cbinfo[info->selection] = qemu_clipboard_info_ref(info); 58684e64d3SMarc-André Lureau } 59684e64d3SMarc-André Lureau 60684e64d3SMarc-André Lureau QemuClipboardInfo *qemu_clipboard_info(QemuClipboardSelection selection) 61684e64d3SMarc-André Lureau { 62684e64d3SMarc-André Lureau assert(selection < QEMU_CLIPBOARD_SELECTION__COUNT); 63684e64d3SMarc-André Lureau 64684e64d3SMarc-André Lureau return cbinfo[selection]; 65660e8d0fSGerd Hoffmann } 66660e8d0fSGerd Hoffmann 67660e8d0fSGerd Hoffmann QemuClipboardInfo *qemu_clipboard_info_new(QemuClipboardPeer *owner, 68660e8d0fSGerd Hoffmann QemuClipboardSelection selection) 69660e8d0fSGerd Hoffmann { 70660e8d0fSGerd Hoffmann QemuClipboardInfo *info = g_new0(QemuClipboardInfo, 1); 71660e8d0fSGerd Hoffmann 72660e8d0fSGerd Hoffmann info->owner = owner; 73660e8d0fSGerd Hoffmann info->selection = selection; 74660e8d0fSGerd Hoffmann info->refcount = 1; 75660e8d0fSGerd Hoffmann 76660e8d0fSGerd Hoffmann return info; 77660e8d0fSGerd Hoffmann } 78660e8d0fSGerd Hoffmann 79660e8d0fSGerd Hoffmann QemuClipboardInfo *qemu_clipboard_info_ref(QemuClipboardInfo *info) 80660e8d0fSGerd Hoffmann { 81660e8d0fSGerd Hoffmann info->refcount++; 82660e8d0fSGerd Hoffmann return info; 83660e8d0fSGerd Hoffmann } 84660e8d0fSGerd Hoffmann 85660e8d0fSGerd Hoffmann void qemu_clipboard_info_unref(QemuClipboardInfo *info) 86660e8d0fSGerd Hoffmann { 87660e8d0fSGerd Hoffmann uint32_t type; 88660e8d0fSGerd Hoffmann 89660e8d0fSGerd Hoffmann if (!info) { 90660e8d0fSGerd Hoffmann return; 91660e8d0fSGerd Hoffmann } 92660e8d0fSGerd Hoffmann 93660e8d0fSGerd Hoffmann info->refcount--; 94660e8d0fSGerd Hoffmann if (info->refcount > 0) { 95660e8d0fSGerd Hoffmann return; 96660e8d0fSGerd Hoffmann } 97660e8d0fSGerd Hoffmann 98660e8d0fSGerd Hoffmann for (type = 0; type < QEMU_CLIPBOARD_TYPE__COUNT; type++) { 99660e8d0fSGerd Hoffmann g_free(info->types[type].data); 100660e8d0fSGerd Hoffmann } 101660e8d0fSGerd Hoffmann g_free(info); 102660e8d0fSGerd Hoffmann } 103660e8d0fSGerd Hoffmann 104660e8d0fSGerd Hoffmann void qemu_clipboard_request(QemuClipboardInfo *info, 105660e8d0fSGerd Hoffmann QemuClipboardType type) 106660e8d0fSGerd Hoffmann { 107660e8d0fSGerd Hoffmann if (info->types[type].data || 108660e8d0fSGerd Hoffmann info->types[type].requested || 109660e8d0fSGerd Hoffmann !info->types[type].available || 110660e8d0fSGerd Hoffmann !info->owner) 111660e8d0fSGerd Hoffmann return; 112660e8d0fSGerd Hoffmann 113660e8d0fSGerd Hoffmann info->types[type].requested = true; 114660e8d0fSGerd Hoffmann info->owner->request(info, type); 115660e8d0fSGerd Hoffmann } 116660e8d0fSGerd Hoffmann 117660e8d0fSGerd Hoffmann void qemu_clipboard_set_data(QemuClipboardPeer *peer, 118660e8d0fSGerd Hoffmann QemuClipboardInfo *info, 119660e8d0fSGerd Hoffmann QemuClipboardType type, 120660e8d0fSGerd Hoffmann uint32_t size, 1217e3e20d8SAkihiko Odaki const void *data, 122660e8d0fSGerd Hoffmann bool update) 123660e8d0fSGerd Hoffmann { 124660e8d0fSGerd Hoffmann if (!info || 125660e8d0fSGerd Hoffmann info->owner != peer) { 126660e8d0fSGerd Hoffmann return; 127660e8d0fSGerd Hoffmann } 128660e8d0fSGerd Hoffmann 129660e8d0fSGerd Hoffmann g_free(info->types[type].data); 130660e8d0fSGerd Hoffmann info->types[type].data = g_memdup(data, size); 131660e8d0fSGerd Hoffmann info->types[type].size = size; 132660e8d0fSGerd Hoffmann info->types[type].available = true; 133660e8d0fSGerd Hoffmann 134660e8d0fSGerd Hoffmann if (update) { 135660e8d0fSGerd Hoffmann qemu_clipboard_update(info); 136660e8d0fSGerd Hoffmann } 137660e8d0fSGerd Hoffmann } 138