VirtualBox

source: vbox/trunk/src/VBox/HostServices/SharedClipboard/x11-clipboard.cpp@ 24777

Last change on this file since 24777 was 24142, checked in by vboxsync, 15 years ago

GuestHost and HostServices/SharedClipboard/x11: fix a crash when clipboard initialisation fails

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 19.9 KB
Line 
1/** @file
2 *
3 * Shared Clipboard:
4 * Linux host.
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#define LOG_GROUP LOG_GROUP_SHARED_CLIPBOARD
24
25#include <string.h>
26
27#include <iprt/assert.h>
28#include <iprt/critsect.h>
29#include <iprt/mem.h>
30#include <iprt/semaphore.h>
31
32#include <VBox/GuestHost/SharedClipboard.h>
33#include <VBox/HostServices/VBoxClipboardSvc.h>
34
35#include "VBoxClipboard.h"
36
37struct _VBOXCLIPBOARDREQFROMVBOX;
38typedef struct _VBOXCLIPBOARDREQFROMVBOX VBOXCLIPBOARDREQFROMVBOX;
39
40/** Global context information used by the host glue for the X11 clipboard
41 * backend */
42struct _VBOXCLIPBOARDCONTEXT
43{
44 /** This mutex is grabbed during any critical operations on the clipboard
45 * which might clash with others. */
46 RTCRITSECT clipboardMutex;
47 /** The currently pending request for data from VBox. NULL if there is
48 * no request pending. The protocol for completing a request is to grab
49 * the critical section, check that @a pReq is not NULL, fill in the data
50 * fields and set @a pReq to NULL. The protocol for cancelling a pending
51 * request is to grab the critical section and set pReq to NULL.
52 * It is an error if a request arrives while another one is pending, and
53 * the backend is responsible for ensuring that this does not happen. */
54 VBOXCLIPBOARDREQFROMVBOX *pReq;
55
56 /** Pointer to the opaque X11 backend structure */
57 CLIPBACKEND *pBackend;
58 /** Pointer to the VBox host client data structure. */
59 VBOXCLIPBOARDCLIENTDATA *pClient;
60 /** We set this when we start shutting down as a hint not to post any new
61 * requests. */
62 bool fShuttingDown;
63};
64
65/**
66 * Report formats available in the X11 clipboard to VBox.
67 * @param pCtx Opaque context pointer for the glue code
68 * @param u32Formats The formats available
69 * @note Host glue code
70 */
71void ClipReportX11Formats(VBOXCLIPBOARDCONTEXT *pCtx,
72 uint32_t u32Formats)
73{
74 LogRelFlowFunc(("called. pCtx=%p, u32Formats=%02X\n", pCtx, u32Formats));
75 vboxSvcClipboardReportMsg(pCtx->pClient,
76 VBOX_SHARED_CLIPBOARD_HOST_MSG_FORMATS,
77 u32Formats);
78}
79
80/**
81 * Initialise the host side of the shared clipboard.
82 * @note Host glue code
83 */
84int vboxClipboardInit (void)
85{
86 return VINF_SUCCESS;
87}
88
89/**
90 * Terminate the host side of the shared clipboard.
91 * @note host glue code
92 */
93void vboxClipboardDestroy (void)
94{
95
96}
97
98/**
99 * Connect a guest to the shared clipboard.
100 * @note host glue code
101 * @note on the host, we assume that some other application already owns
102 * the clipboard and leave ownership to X11.
103 */
104int vboxClipboardConnect (VBOXCLIPBOARDCLIENTDATA *pClient)
105{
106 int rc = VINF_SUCCESS;
107 CLIPBACKEND *pBackend = NULL;
108
109 LogRel(("Starting host clipboard service\n"));
110 VBOXCLIPBOARDCONTEXT *pCtx =
111 (VBOXCLIPBOARDCONTEXT *) RTMemAllocZ(sizeof(VBOXCLIPBOARDCONTEXT));
112 if (!pCtx)
113 rc = VERR_NO_MEMORY;
114 else
115 {
116 RTCritSectInit(&pCtx->clipboardMutex);
117 pBackend = ClipConstructX11(pCtx);
118 if (pBackend == NULL)
119 rc = VERR_NO_MEMORY;
120 else
121 {
122 pCtx->pBackend = pBackend;
123 pClient->pCtx = pCtx;
124 pCtx->pClient = pClient;
125 rc = ClipStartX11(pBackend, true /* grab shared clipboard */);
126 }
127 if (RT_FAILURE(rc))
128 RTCritSectDelete(&pCtx->clipboardMutex);
129 }
130 if (RT_FAILURE(rc))
131 {
132 RTMemFree(pCtx);
133 LogRel(("Failed to initialise the shared clipboard\n"));
134 }
135 LogRelFlowFunc(("returning %Rrc\n", rc));
136 return rc;
137}
138
139/**
140 * Synchronise the contents of the host clipboard with the guest, called
141 * after a save and restore of the guest.
142 * @note Host glue code
143 */
144int vboxClipboardSync (VBOXCLIPBOARDCLIENTDATA *pClient)
145{
146 /* Tell the guest we have no data in case X11 is not available. If
147 * there is data in the host clipboard it will automatically be sent to
148 * the guest when the clipboard starts up. */
149 vboxSvcClipboardReportMsg (pClient,
150 VBOX_SHARED_CLIPBOARD_HOST_MSG_FORMATS, 0);
151 return VINF_SUCCESS;
152}
153
154/**
155 * Shut down the shared clipboard service and "disconnect" the guest.
156 * @note Host glue code
157 */
158void vboxClipboardDisconnect (VBOXCLIPBOARDCLIENTDATA *pClient)
159{
160 LogRelFlow(("vboxClipboardDisconnect\n"));
161
162 LogRel(("Stopping the host clipboard service\n"));
163 VBOXCLIPBOARDCONTEXT *pCtx = pClient->pCtx;
164 /* Drop the reference to the client, in case it is still there. This
165 * will cause any outstanding clipboard data requests from X11 to fail
166 * immediately. */
167 pCtx->fShuttingDown = true;
168 /* If there is a currently pending request, release it immediately. */
169 vboxClipboardWriteData(pClient, NULL, 0, 0);
170 int rc = ClipStopX11(pCtx->pBackend);
171 /** @todo handle this slightly more reasonably, or be really sure
172 * it won't go wrong. */
173 AssertRC(rc);
174 if (RT_SUCCESS(rc)) /* And if not? */
175 {
176 ClipDestructX11(pCtx->pBackend);
177 RTCritSectDelete(&pCtx->clipboardMutex);
178 RTMemFree(pCtx);
179 }
180}
181
182/**
183 * VBox is taking possession of the shared clipboard.
184 *
185 * @param pClient Context data for the guest system
186 * @param u32Formats Clipboard formats the guest is offering
187 * @note Host glue code
188 */
189void vboxClipboardFormatAnnounce (VBOXCLIPBOARDCLIENTDATA *pClient,
190 uint32_t u32Formats)
191{
192 LogRelFlowFunc(("called. pClient=%p, u32Formats=%02X\n", pClient,
193 u32Formats));
194 ClipAnnounceFormatToX11 (pClient->pCtx->pBackend, u32Formats);
195}
196
197/** Structure describing a request for clipoard data from the guest. */
198struct _CLIPREADCBREQ
199{
200 /** Where to write the returned data to. */
201 void *pv;
202 /** The size of the buffer in pv */
203 uint32_t cb;
204 /** The actual size of the data written */
205 uint32_t *pcbActual;
206};
207
208/**
209 * Called when VBox wants to read the X11 clipboard.
210 *
211 * @returns VINF_SUCCESS on successful completion
212 * @returns VINF_HGCM_ASYNC_EXECUTE if the operation will complete
213 * asynchronously
214 * @returns iprt status code on failure
215 * @param pClient Context information about the guest VM
216 * @param u32Format The format that the guest would like to receive the data in
217 * @param pv Where to write the data to
218 * @param cb The size of the buffer to write the data to
219 * @param pcbActual Where to write the actual size of the written data
220 * @note We always fail or complete asynchronously
221 * @note On success allocates a CLIPREADCBREQ structure which must be
222 * freed in ClipCompleteDataRequestFromX11 when it is called back from
223 * the backend code.
224 *
225 */
226int vboxClipboardReadData (VBOXCLIPBOARDCLIENTDATA *pClient,
227 uint32_t u32Format, void *pv, uint32_t cb,
228 uint32_t *pcbActual)
229{
230 LogRelFlowFunc(("pClient=%p, u32Format=%02X, pv=%p, cb=%u, pcbActual=%p",
231 pClient, u32Format, pv, cb, pcbActual));
232
233 int rc = VINF_SUCCESS;
234 CLIPREADCBREQ *pReq = (CLIPREADCBREQ *) RTMemAlloc(sizeof(CLIPREADCBREQ));
235 if (!pReq)
236 rc = VERR_NO_MEMORY;
237 else
238 {
239 pReq->pv = pv;
240 pReq->cb = cb;
241 pReq->pcbActual = pcbActual;
242 rc = ClipRequestDataFromX11(pClient->pCtx->pBackend, u32Format, pReq);
243 if (RT_SUCCESS(rc))
244 rc = VINF_HGCM_ASYNC_EXECUTE;
245 }
246 LogRelFlowFunc(("returning %Rrc\n", rc));
247 return rc;
248}
249
250/**
251 * Complete a request from VBox for the X11 clipboard data. The data should
252 * be written to the buffer provided in the initial request.
253 * @param pCtx request context information
254 * @param rc the completion status of the request
255 * @param cbActual on successful completion, the number of bytes of data
256 * actually written, on buffer overflow the size of the
257 * buffer needed, ignored otherwise
258 * @todo change this to deal with the buffer issues rather than offloading
259 * them onto the caller
260 */
261void ClipCompleteDataRequestFromX11(VBOXCLIPBOARDCONTEXT *pCtx, int rc,
262 CLIPREADCBREQ *pReq, void *pv,
263 uint32_t cb)
264{
265 if (cb <= pReq->cb)
266 memcpy(pReq->pv, pv, cb);
267 RTMemFree(pReq);
268 vboxSvcClipboardCompleteReadData(pCtx->pClient, rc, cb);
269}
270
271/** A request for clipboard data from VBox */
272struct _VBOXCLIPBOARDREQFROMVBOX
273{
274 /** Data received */
275 void *pv;
276 /** The size of the data */
277 uint32_t cb;
278 /** Format of the data */
279 uint32_t format;
280 /** A semaphore for waiting for the data */
281 RTSEMEVENT finished;
282};
283
284/** Wait for clipboard data requested from VBox to arrive. */
285static int clipWaitForDataFromVBox(VBOXCLIPBOARDCONTEXT *pCtx,
286 VBOXCLIPBOARDREQFROMVBOX *pReq,
287 uint32_t u32Format)
288{
289 int rc = VINF_SUCCESS;
290 LogRelFlowFunc(("pCtx=%p, pReq=%p, u32Format=%02X\n", pCtx, pReq, u32Format));
291 /* Request data from VBox */
292 vboxSvcClipboardReportMsg(pCtx->pClient,
293 VBOX_SHARED_CLIPBOARD_HOST_MSG_READ_DATA,
294 u32Format);
295 /* Which will signal us when it is ready. We use a timeout here
296 * because we can't be sure that the guest will behave correctly.
297 */
298 rc = RTSemEventWait(pReq->finished, CLIPBOARD_TIMEOUT);
299 /* If the request hasn't yet completed then we cancel it. We use
300 * the critical section to prevent these operations colliding. */
301 RTCritSectEnter(&pCtx->clipboardMutex);
302 /* The data may have arrived between the semaphore timing out and
303 * our grabbing the mutex. */
304 if (rc == VERR_TIMEOUT && pReq->pv != NULL)
305 rc = VINF_SUCCESS;
306 if (pCtx->pReq == pReq)
307 pCtx->pReq = NULL;
308 Assert(pCtx->pReq == NULL);
309 RTCritSectLeave(&pCtx->clipboardMutex);
310 if (RT_SUCCESS(rc) && (pReq->pv == NULL))
311 rc = VERR_NO_DATA;
312 LogRelFlowFunc(("returning %Rrc\n", rc));
313 return rc;
314}
315
316/** Post a request for clipboard data to VBox/the guest and wait for it to be
317 * completed. */
318static int clipRequestDataFromVBox(VBOXCLIPBOARDCONTEXT *pCtx,
319 VBOXCLIPBOARDREQFROMVBOX *pReq,
320 uint32_t u32Format)
321{
322 int rc = VINF_SUCCESS;
323 LogRelFlowFunc(("pCtx=%p, pReq=%p, u32Format=%02X\n", pCtx, pReq,
324 u32Format));
325 /* Start by "posting" the request for the next invocation of
326 * vboxClipboardWriteData. */
327 RTCritSectEnter(&pCtx->clipboardMutex);
328 if (pCtx->pReq != NULL)
329 {
330 /* This would be a violation of the protocol, see the comments in the
331 * context structure definition. */
332 Assert(false);
333 rc = VERR_WRONG_ORDER;
334 }
335 else
336 pCtx->pReq = pReq;
337 RTCritSectLeave(&pCtx->clipboardMutex);
338 if (RT_SUCCESS(rc))
339 rc = clipWaitForDataFromVBox(pCtx, pReq, u32Format);
340 LogRelFlowFunc(("returning %Rrc\n", rc));
341 return rc;
342}
343
344/**
345 * Send a request to VBox to transfer the contents of its clipboard to X11.
346 *
347 * @param pCtx Pointer to the host clipboard structure
348 * @param u32Format The format in which the data should be transfered
349 * @param ppv On success and if pcb > 0, this will point to a buffer
350 * to be freed with RTMemFree containing the data read.
351 * @param pcb On success, this contains the number of bytes of data
352 * returned
353 * @note Host glue code.
354 */
355int ClipRequestDataForX11 (VBOXCLIPBOARDCONTEXT *pCtx,
356 uint32_t u32Format, void **ppv,
357 uint32_t *pcb)
358{
359 VBOXCLIPBOARDREQFROMVBOX request = { NULL };
360
361 LogRelFlowFunc(("pCtx=%p, u32Format=%02X, ppv=%p, pcb=%p\n", pCtx,
362 u32Format, ppv, pcb));
363 if (pCtx->fShuttingDown)
364 {
365 /* The shared clipboard is disconnecting. */
366 LogRelFunc(("host requested guest clipboard data after guest had disconnected.\n"));
367 return VERR_WRONG_ORDER;
368 }
369 int rc = RTSemEventCreate(&request.finished);
370 if (RT_SUCCESS(rc))
371 {
372 rc = clipRequestDataFromVBox(pCtx, &request, u32Format);
373 RTSemEventDestroy(request.finished);
374 }
375 if (RT_SUCCESS(rc))
376 {
377 *ppv = request.pv;
378 *pcb = request.cb;
379 }
380 LogRelFlowFunc(("returning %Rrc\n", rc));
381 if (RT_SUCCESS(rc))
382 LogRelFlowFunc(("*ppv=%.*ls, *pcb=%u\n", *pcb / 2, *ppv, *pcb));
383 return rc;
384}
385
386/**
387 * Called when we have requested data from VBox and that data has arrived.
388 *
389 * @param pClient Context information about the guest VM
390 * @param pv Buffer to which the data was written
391 * @param cb The size of the data written
392 * @param u32Format The format of the data written
393 * @note Host glue code
394 */
395void vboxClipboardWriteData (VBOXCLIPBOARDCLIENTDATA *pClient,
396 void *pv, uint32_t cb, uint32_t u32Format)
397{
398 LogRelFlowFunc (("called. pClient=%p, pv=%p (%.*ls), cb=%u, u32Format=%02X\n",
399 pClient, pv, cb / 2, pv, cb, u32Format));
400
401 VBOXCLIPBOARDCONTEXT *pCtx = pClient->pCtx;
402 /* Grab the mutex and check whether there is a pending request for data.
403 */
404 RTCritSectEnter(&pCtx->clipboardMutex);
405 VBOXCLIPBOARDREQFROMVBOX *pReq = pCtx->pReq;
406 if (pReq != NULL)
407 {
408 if (cb > 0)
409 {
410 pReq->pv = RTMemDup(pv, cb);
411 if (pReq->pv != NULL) /* NULL may also mean no memory... */
412 {
413 pReq->cb = cb;
414 pReq->format = u32Format;
415 }
416 }
417 /* Signal that the request has been completed. */
418 RTSemEventSignal(pReq->finished);
419 pCtx->pReq = NULL;
420 }
421 RTCritSectLeave(&pCtx->clipboardMutex);
422}
423
424#ifdef TESTCASE
425#include <iprt/initterm.h>
426#include <iprt/stream.h>
427
428#define TEST_NAME "tstClipboardX11-2"
429
430struct _CLIPBACKEND
431{
432 uint32_t formats;
433 struct _READDATA
434 {
435 uint32_t format;
436 int rc;
437 CLIPREADCBREQ *pReq;
438 } readData;
439 struct _COMPLETEREAD
440 {
441 int rc;
442 uint32_t cbActual;
443 } completeRead;
444 struct _WRITEDATA
445 {
446 void *pv;
447 uint32_t cb;
448 uint32_t format;
449 bool timeout;
450 } writeData;
451 struct _REPORTDATA
452 {
453 uint32_t format;
454 } reportData;
455};
456
457void vboxSvcClipboardReportMsg (VBOXCLIPBOARDCLIENTDATA *pClient, uint32_t u32Msg, uint32_t u32Formats)
458{
459 CLIPBACKEND *pBackend = pClient->pCtx->pBackend;
460 if ( (u32Msg == VBOX_SHARED_CLIPBOARD_HOST_MSG_READ_DATA)
461 && !pBackend->writeData.timeout)
462 vboxClipboardWriteData(pClient, pBackend->writeData.pv,
463 pBackend->writeData.cb,
464 pBackend->writeData.format);
465 else
466 return;
467}
468
469void vboxSvcClipboardCompleteReadData(VBOXCLIPBOARDCLIENTDATA *pClient, int rc, uint32_t cbActual)
470{
471 CLIPBACKEND *pBackend = pClient->pCtx->pBackend;
472 pBackend->completeRead.rc = rc;
473 pBackend->completeRead.cbActual = cbActual;
474}
475
476CLIPBACKEND *ClipConstructX11(VBOXCLIPBOARDCONTEXT *pFrontend)
477{
478 return (CLIPBACKEND *)RTMemAllocZ(sizeof(CLIPBACKEND));
479}
480
481void ClipDestructX11(CLIPBACKEND *pBackend)
482{
483 RTMemFree(pBackend);
484}
485
486int ClipStartX11(CLIPBACKEND *pBackend, bool)
487{
488 return VINF_SUCCESS;
489}
490
491int ClipStopX11(CLIPBACKEND *pBackend)
492{
493 return VINF_SUCCESS;
494}
495
496void ClipAnnounceFormatToX11(CLIPBACKEND *pBackend,
497 uint32_t u32Formats)
498{
499 pBackend->formats = u32Formats;
500}
501
502extern int ClipRequestDataFromX11(CLIPBACKEND *pBackend, uint32_t u32Format,
503 CLIPREADCBREQ *pReq)
504{
505 pBackend->readData.format = u32Format;
506 pBackend->readData.pReq = pReq;
507 return pBackend->readData.rc;
508}
509
510int main()
511{
512 VBOXCLIPBOARDCLIENTDATA client;
513 unsigned cErrors = 0;
514 int rc = RTR3Init();
515 RTPrintf(TEST_NAME ": TESTING\n");
516 AssertRCReturn(rc, 1);
517 rc = vboxClipboardConnect(&client);
518 CLIPBACKEND *pBackend = client.pCtx->pBackend;
519 AssertRCReturn(rc, 1);
520 vboxClipboardFormatAnnounce(&client,
521 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
522 if (pBackend->formats != VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
523 {
524 RTPrintf(TEST_NAME ": vboxClipboardFormatAnnounce failed with VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT\n");
525 ++cErrors;
526 }
527 pBackend->readData.rc = VINF_SUCCESS;
528 client.asyncRead.callHandle = (VBOXHGCMCALLHANDLE)pBackend;
529 client.asyncRead.paParms = (VBOXHGCMSVCPARM *)&client;
530 uint32_t u32Dummy;
531 rc = vboxClipboardReadData(&client, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT,
532 &u32Dummy, 42, &u32Dummy);
533 if (rc != VINF_HGCM_ASYNC_EXECUTE)
534 {
535 RTPrintf(TEST_NAME ": vboxClipboardReadData returned %Rrc\n", rc);
536 ++cErrors;
537 }
538 else
539 {
540 if ( pBackend->readData.format !=
541 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT
542 || pBackend->readData.pReq->pv != &u32Dummy
543 || pBackend->readData.pReq->cb != 42
544 || pBackend->readData.pReq->pcbActual != &u32Dummy)
545 {
546 RTPrintf(TEST_NAME ": format=%u, pReq->pv=%p, pReq->cb=%u, pReq->pcbActual=%p\n",
547 pBackend->readData.format, pBackend->readData.pReq->pv,
548 pBackend->readData.pReq->cb,
549 pBackend->readData.pReq->pcbActual);
550 ++cErrors;
551 }
552 else
553 {
554 ClipCompleteDataRequestFromX11(client.pCtx, VERR_NO_DATA,
555 pBackend->readData.pReq, NULL, 43);
556 if ( pBackend->completeRead.rc != VERR_NO_DATA
557 || pBackend->completeRead.cbActual != 43)
558 {
559 RTPrintf(TEST_NAME ": rc=%Rrc, cbActual=%u\n",
560 pBackend->completeRead.rc,
561 pBackend->completeRead.cbActual);
562 ++cErrors;
563 }
564 }
565 }
566 void *pv;
567 uint32_t cb;
568 pBackend->writeData.pv = (void *)"testing";
569 pBackend->writeData.cb = sizeof("testing");
570 pBackend->writeData.format = 1234;
571 pBackend->reportData.format = 4321; /* XX this should be handled! */
572 rc = ClipRequestDataForX11(client.pCtx, 23, &pv, &cb);
573 if ( rc != VINF_SUCCESS
574 || strcmp((const char *)pv, "testing") != 0
575 || cb != sizeof("testing"))
576 {
577 RTPrintf("rc=%Rrc, pv=%p, cb=%u\n", rc, pv, cb);
578 ++cErrors;
579 }
580 else
581 RTMemFree(pv);
582 pBackend->writeData.timeout = true;
583 rc = ClipRequestDataForX11(client.pCtx, 23, &pv, &cb);
584 if (rc != VERR_TIMEOUT)
585 {
586 RTPrintf("rc=%Rrc, expected VERR_TIMEOUT\n", rc);
587 ++cErrors;
588 }
589 pBackend->writeData.pv = NULL;
590 pBackend->writeData.cb = 0;
591 pBackend->writeData.timeout = false;
592 rc = ClipRequestDataForX11(client.pCtx, 23, &pv, &cb);
593 if (rc != VERR_NO_DATA)
594 {
595 RTPrintf("rc=%Rrc, expected VERR_NO_DATA\n", rc);
596 ++cErrors;
597 }
598 /* Data arriving after a timeout should *not* cause any segfaults or
599 * memory leaks. Check with Valgrind! */
600 vboxClipboardWriteData(&client, (void *)"tested", sizeof("tested"), 999);
601 vboxClipboardDisconnect(&client);
602 if (cErrors > 0)
603 RTPrintf(TEST_NAME ": errors: %u\n", cErrors);
604 return cErrors > 0 ? 1 : 0;
605}
606#endif /* TESTCASE */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette