xref: /openbmc/qemu/ui/clipboard.c (revision 1b17f1e9f962f5ae9cd559d7f23718ceed71b813)
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, &notify);
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