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 { 11660e8d0fSGerd Hoffmann notifier_list_add(&clipboard_notifiers, &peer->update); 12660e8d0fSGerd Hoffmann } 13660e8d0fSGerd Hoffmann 14660e8d0fSGerd Hoffmann void qemu_clipboard_peer_unregister(QemuClipboardPeer *peer) 15660e8d0fSGerd Hoffmann { 16660e8d0fSGerd Hoffmann notifier_remove(&peer->update); 17660e8d0fSGerd Hoffmann } 18660e8d0fSGerd Hoffmann 19482bbaf4SMarc-André Lureau bool qemu_clipboard_peer_owns(QemuClipboardPeer *peer, 20482bbaf4SMarc-André Lureau QemuClipboardSelection selection) 21482bbaf4SMarc-André Lureau { 22482bbaf4SMarc-André Lureau QemuClipboardInfo *info = qemu_clipboard_info(selection); 23482bbaf4SMarc-André Lureau 24482bbaf4SMarc-André Lureau return info && info->owner == peer; 25482bbaf4SMarc-André Lureau } 26482bbaf4SMarc-André Lureau 27*1387865eSMarc-André Lureau void qemu_clipboard_peer_release(QemuClipboardPeer *peer, 28*1387865eSMarc-André Lureau QemuClipboardSelection selection) 29*1387865eSMarc-André Lureau { 30*1387865eSMarc-André Lureau g_autoptr(QemuClipboardInfo) info = NULL; 31*1387865eSMarc-André Lureau 32*1387865eSMarc-André Lureau if (qemu_clipboard_peer_owns(peer, selection)) { 33*1387865eSMarc-André Lureau /* set empty clipboard info */ 34*1387865eSMarc-André Lureau info = qemu_clipboard_info_new(NULL, selection); 35*1387865eSMarc-André Lureau qemu_clipboard_update(info); 36*1387865eSMarc-André Lureau } 37*1387865eSMarc-André Lureau } 38*1387865eSMarc-André Lureau 39660e8d0fSGerd Hoffmann void qemu_clipboard_update(QemuClipboardInfo *info) 40660e8d0fSGerd Hoffmann { 41684e64d3SMarc-André Lureau g_autoptr(QemuClipboardInfo) old = NULL; 42684e64d3SMarc-André Lureau assert(info->selection < QEMU_CLIPBOARD_SELECTION__COUNT); 43684e64d3SMarc-André Lureau 44660e8d0fSGerd Hoffmann notifier_list_notify(&clipboard_notifiers, info); 45684e64d3SMarc-André Lureau 46684e64d3SMarc-André Lureau old = cbinfo[info->selection]; 47684e64d3SMarc-André Lureau cbinfo[info->selection] = qemu_clipboard_info_ref(info); 48684e64d3SMarc-André Lureau } 49684e64d3SMarc-André Lureau 50684e64d3SMarc-André Lureau QemuClipboardInfo *qemu_clipboard_info(QemuClipboardSelection selection) 51684e64d3SMarc-André Lureau { 52684e64d3SMarc-André Lureau assert(selection < QEMU_CLIPBOARD_SELECTION__COUNT); 53684e64d3SMarc-André Lureau 54684e64d3SMarc-André Lureau return cbinfo[selection]; 55660e8d0fSGerd Hoffmann } 56660e8d0fSGerd Hoffmann 57660e8d0fSGerd Hoffmann QemuClipboardInfo *qemu_clipboard_info_new(QemuClipboardPeer *owner, 58660e8d0fSGerd Hoffmann QemuClipboardSelection selection) 59660e8d0fSGerd Hoffmann { 60660e8d0fSGerd Hoffmann QemuClipboardInfo *info = g_new0(QemuClipboardInfo, 1); 61660e8d0fSGerd Hoffmann 62660e8d0fSGerd Hoffmann info->owner = owner; 63660e8d0fSGerd Hoffmann info->selection = selection; 64660e8d0fSGerd Hoffmann info->refcount = 1; 65660e8d0fSGerd Hoffmann 66660e8d0fSGerd Hoffmann return info; 67660e8d0fSGerd Hoffmann } 68660e8d0fSGerd Hoffmann 69660e8d0fSGerd Hoffmann QemuClipboardInfo *qemu_clipboard_info_ref(QemuClipboardInfo *info) 70660e8d0fSGerd Hoffmann { 71660e8d0fSGerd Hoffmann info->refcount++; 72660e8d0fSGerd Hoffmann return info; 73660e8d0fSGerd Hoffmann } 74660e8d0fSGerd Hoffmann 75660e8d0fSGerd Hoffmann void qemu_clipboard_info_unref(QemuClipboardInfo *info) 76660e8d0fSGerd Hoffmann { 77660e8d0fSGerd Hoffmann uint32_t type; 78660e8d0fSGerd Hoffmann 79660e8d0fSGerd Hoffmann if (!info) { 80660e8d0fSGerd Hoffmann return; 81660e8d0fSGerd Hoffmann } 82660e8d0fSGerd Hoffmann 83660e8d0fSGerd Hoffmann info->refcount--; 84660e8d0fSGerd Hoffmann if (info->refcount > 0) { 85660e8d0fSGerd Hoffmann return; 86660e8d0fSGerd Hoffmann } 87660e8d0fSGerd Hoffmann 88660e8d0fSGerd Hoffmann for (type = 0; type < QEMU_CLIPBOARD_TYPE__COUNT; type++) { 89660e8d0fSGerd Hoffmann g_free(info->types[type].data); 90660e8d0fSGerd Hoffmann } 91660e8d0fSGerd Hoffmann g_free(info); 92660e8d0fSGerd Hoffmann } 93660e8d0fSGerd Hoffmann 94660e8d0fSGerd Hoffmann void qemu_clipboard_request(QemuClipboardInfo *info, 95660e8d0fSGerd Hoffmann QemuClipboardType type) 96660e8d0fSGerd Hoffmann { 97660e8d0fSGerd Hoffmann if (info->types[type].data || 98660e8d0fSGerd Hoffmann info->types[type].requested || 99660e8d0fSGerd Hoffmann !info->types[type].available || 100660e8d0fSGerd Hoffmann !info->owner) 101660e8d0fSGerd Hoffmann return; 102660e8d0fSGerd Hoffmann 103660e8d0fSGerd Hoffmann info->types[type].requested = true; 104660e8d0fSGerd Hoffmann info->owner->request(info, type); 105660e8d0fSGerd Hoffmann } 106660e8d0fSGerd Hoffmann 107660e8d0fSGerd Hoffmann void qemu_clipboard_set_data(QemuClipboardPeer *peer, 108660e8d0fSGerd Hoffmann QemuClipboardInfo *info, 109660e8d0fSGerd Hoffmann QemuClipboardType type, 110660e8d0fSGerd Hoffmann uint32_t size, 1117e3e20d8SAkihiko Odaki const void *data, 112660e8d0fSGerd Hoffmann bool update) 113660e8d0fSGerd Hoffmann { 114660e8d0fSGerd Hoffmann if (!info || 115660e8d0fSGerd Hoffmann info->owner != peer) { 116660e8d0fSGerd Hoffmann return; 117660e8d0fSGerd Hoffmann } 118660e8d0fSGerd Hoffmann 119660e8d0fSGerd Hoffmann g_free(info->types[type].data); 120660e8d0fSGerd Hoffmann info->types[type].data = g_memdup(data, size); 121660e8d0fSGerd Hoffmann info->types[type].size = size; 122660e8d0fSGerd Hoffmann info->types[type].available = true; 123660e8d0fSGerd Hoffmann 124660e8d0fSGerd Hoffmann if (update) { 125660e8d0fSGerd Hoffmann qemu_clipboard_update(info); 126660e8d0fSGerd Hoffmann } 127660e8d0fSGerd Hoffmann } 128