1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-03 12:11:33 +01:00

scd: Introduce a virtual reader table.

The vreader table makes the code more clear by explicitly talking
about APDU slots and reader indices.  It also accommodates for future
extensions.

* scd/scdaemon.h (server_control_s): Remove READER_SLOT.
* scd/scdaemon.c (scd_init_default_ctrl): Do not init READER_SLOT.
* scd/app.c (check_application_conflict): Add arg SLOT.
* scd/command.c (slot_status_s): Rename to vreader_s.
(server_local_s): Add field VREADER_IDX as replacement for
the READER_SLOT in server_control_s.  Change all users.
(slot_table): Rename to vreader_table.  Change all users.
(vreader_slot): New.
(do_reset, cmd_apdu): Map vreader to apdu slot.
(get_reader_slot): Rename to get_current_reader.  Return -1 on error.
(open_card): Map vreader toapdu slot.  Pass slot to
check_application_conflict.
(scd_command_handler): Init VREADER_IDX.
(update_reader_status_file): Reset SLOT field on error.
This commit is contained in:
Werner Koch 2011-12-13 16:55:42 +01:00
parent 1116466278
commit 24e121ef26
5 changed files with 129 additions and 99 deletions

View File

@ -143,7 +143,8 @@ size_t app_help_read_length_of_cert (int slot, int fid, size_t *r_certoff);
/*-- app.c --*/ /*-- app.c --*/
void app_dump_state (void); void app_dump_state (void);
void application_notify_card_reset (int slot); void application_notify_card_reset (int slot);
gpg_error_t check_application_conflict (ctrl_t ctrl, const char *name); gpg_error_t check_application_conflict (ctrl_t ctrl, int slot,
const char *name);
gpg_error_t select_application (ctrl_t ctrl, int slot, const char *name, gpg_error_t select_application (ctrl_t ctrl, int slot, const char *name,
app_t *r_app); app_t *r_app);
char *get_supported_applications (void); char *get_supported_applications (void);

View File

@ -213,11 +213,12 @@ application_notify_card_reset (int slot)
used to request a specific application and the connection has used to request a specific application and the connection has
already done a select_application. */ already done a select_application. */
gpg_error_t gpg_error_t
check_application_conflict (ctrl_t ctrl, const char *name) check_application_conflict (ctrl_t ctrl, int slot, const char *name)
{ {
int slot = ctrl->reader_slot;
app_t app; app_t app;
(void)ctrl;
if (slot < 0 || slot >= DIM (lock_table)) if (slot < 0 || slot >= DIM (lock_table))
return gpg_error (GPG_ERR_INV_VALUE); return gpg_error (GPG_ERR_INV_VALUE);

View File

@ -1,6 +1,6 @@
/* command.c - SCdaemon command handler /* command.c - SCdaemon command handler
* Copyright (C) 2001, 2002, 2003, 2004, 2005, * Copyright (C) 2001, 2002, 2003, 2004, 2005,
* 2007, 2008, 2009 Free Software Foundation, Inc. * 2007, 2008, 2009, 2011 Free Software Foundation, Inc.
* *
* This file is part of GnuPG. * This file is part of GnuPG.
* *
@ -62,32 +62,37 @@
|| gpg_err_code (_r) == GPG_ERR_CARD_REMOVED \ || gpg_err_code (_r) == GPG_ERR_CARD_REMOVED \
|| gpg_err_code (_r) == GPG_ERR_CARD_RESET \ || gpg_err_code (_r) == GPG_ERR_CARD_RESET \
|| gpg_err_code (_r) == GPG_ERR_ENODEV ) \ || gpg_err_code (_r) == GPG_ERR_ENODEV ) \
update_card_removed ((c)->reader_slot, 1); \ update_card_removed ((c)->server_local->vreader_idx, 1); \
} while (0) } while (0)
#define IS_LOCKED(c) \ #define IS_LOCKED(c) \
(locked_session && locked_session != (c)->server_local \ (locked_session \
&& (c)->reader_slot != -1 && locked_session->ctrl_backlink \ && locked_session != (c)->server_local \
&& (c)->reader_slot == locked_session->ctrl_backlink->reader_slot) && (c)->server_local->vreader_idx != -1 \
&& locked_session->ctrl_backlink \
&& ((c)->server_local->vreader_idx \
== locked_session->ctrl_backlink->server_local->vreader_idx))
/* Flag indicating that the reader has been disabled. */ /* Flag indicating that the reader has been disabled. */
static int reader_disabled; static int reader_disabled;
/* This structure is used to keep track of open readers (slots). */ /* This structure is used to keep track of user readers. To
struct slot_status_s eventually accommodate this structure for RFID cards, where more
than one card is used per reader, we name it virtual reader. */
struct vreader_s
{ {
int valid; /* True if the other objects are valid. */ int valid; /* True if the other objects are valid. */
int slot; /* Slot number of the reader or -1 if not open. */ int slot; /* APDU slot number of the reader or -1 if not open. */
int reset_failed; /* A reset failed. */ int reset_failed; /* A reset failed. */
int any; /* Flag indicating whether any status check has been int any; /* Flag indicating whether any status check has been
done. This is set once to indicate that the status done. This is set once to indicate that the status
tracking for the slot has been initialized. */ tracking for the slot has been initialized. */
unsigned int status; /* Last status of the slot. */ unsigned int status; /* Last status of the reader. */
unsigned int changed; /* Last change counter of the slot. */ unsigned int changed; /* Last change counter of the reader. */
}; };
@ -114,6 +119,9 @@ struct server_local_s
int event_signal; /* Or 0 if not used. */ int event_signal; /* Or 0 if not used. */
#endif #endif
/* Index into the vreader table (command.c) or -1 if not open. */
int vreader_idx;
/* True if the card has been removed and a reset is required to /* True if the card has been removed and a reset is required to
continue operation. */ continue operation. */
int card_removed; int card_removed;
@ -132,10 +140,8 @@ struct server_local_s
}; };
/* The table with information on all used slots. FIXME: This is a /* The table with information on all used virtual readers. */
different slot number than the one used by the APDU layer, and static struct vreader_s vreader_table[10];
should be renamed. */
static struct slot_status_s slot_table[10];
/* To keep track of all running sessions, we link all active server /* To keep track of all running sessions, we link all active server
@ -174,25 +180,37 @@ initialize_module_command (void)
} }
/* Update the CARD_REMOVED element of all sessions using the reader /* Update the CARD_REMOVED element of all sessions using the virtual
given by SLOT to VALUE. */ reader given by VRDR to VALUE. */
static void static void
update_card_removed (int slot, int value) update_card_removed (int vrdr, int value)
{ {
struct server_local_s *sl; struct server_local_s *sl;
for (sl=session_list; sl; sl = sl->next_session) for (sl=session_list; sl; sl = sl->next_session)
if (sl->ctrl_backlink if (sl->ctrl_backlink
&& sl->ctrl_backlink->reader_slot == slot) && sl->ctrl_backlink->server_local->vreader_idx == vrdr)
{ {
sl->card_removed = value; sl->card_removed = value;
} }
/* Let the card application layer know about the removal. */ /* Let the card application layer know about the removal. */
if (value) if (value)
application_notify_card_reset (slot); application_notify_card_reset (vrdr);
} }
/* Helper to return the slot number for a given virtual reader index
VRDR. In case on an error -1 is returned. */
static int
vreader_slot (int vrdr)
{
if (vrdr == -1 || !(vrdr >= 0 && vrdr < DIM(vreader_table)))
return -1;
if (!vreader_table [vrdr].valid)
return -1;
return vreader_table[vrdr].slot;
}
/* Check whether the option NAME appears in LINE. Returns 1 or 0. */ /* Check whether the option NAME appears in LINE. Returns 1 or 0. */
static int static int
@ -279,9 +297,9 @@ hex_to_buffer (const char *string, size_t *r_length)
static void static void
do_reset (ctrl_t ctrl, int send_reset) do_reset (ctrl_t ctrl, int send_reset)
{ {
int slot = ctrl->reader_slot; int vrdr = ctrl->server_local->vreader_idx;
if (!(slot == -1 || (slot >= 0 && slot < DIM(slot_table)))) if (!(vrdr == -1 || (vrdr >= 0 && vrdr < DIM(vreader_table))))
BUG (); BUG ();
/* If there is an active application, release it. Tell all other /* If there is an active application, release it. Tell all other
@ -297,7 +315,7 @@ do_reset (ctrl_t ctrl, int send_reset)
for (sl=session_list; sl; sl = sl->next_session) for (sl=session_list; sl; sl = sl->next_session)
if (sl->ctrl_backlink if (sl->ctrl_backlink
&& sl->ctrl_backlink->reader_slot == slot) && sl->ctrl_backlink->server_local->vreader_idx == vrdr)
{ {
sl->app_ctx_marked_for_release = 1; sl->app_ctx_marked_for_release = 1;
} }
@ -306,13 +324,13 @@ do_reset (ctrl_t ctrl, int send_reset)
/* If we want a real reset for the card, send the reset APDU and /* If we want a real reset for the card, send the reset APDU and
tell the application layer about it. */ tell the application layer about it. */
if (slot != -1 && send_reset && !IS_LOCKED (ctrl) ) if (vrdr != -1 && send_reset && !IS_LOCKED (ctrl) )
{ {
if (apdu_reset (slot)) if (apdu_reset (vreader_table[vrdr].slot))
{ {
slot_table[slot].valid = 0; vreader_table[vrdr].valid = 0;
} }
application_notify_card_reset (slot); application_notify_card_reset (vrdr);
} }
/* If we hold a lock, unlock now. */ /* If we hold a lock, unlock now. */
@ -325,21 +343,21 @@ do_reset (ctrl_t ctrl, int send_reset)
/* Reset the card removed flag for the current reader. We need to /* Reset the card removed flag for the current reader. We need to
take the lock here so that the ticker thread won't concurrently take the lock here so that the ticker thread won't concurrently
try to update the file. Calling update_reader_status_file is try to update the file. Calling update_reader_status_file is
required to get hold of the new status of the card in the slot required to get hold of the new status of the card in the vreader
table. */ table. */
if (!pth_mutex_acquire (&status_file_update_lock, 0, NULL)) if (!pth_mutex_acquire (&status_file_update_lock, 0, NULL))
{ {
log_error ("failed to acquire status_fle_update lock\n"); log_error ("failed to acquire status_file_update lock\n");
ctrl->reader_slot = -1; ctrl->server_local->vreader_idx = -1;
return; return;
} }
update_reader_status_file (0); /* Update slot status table. */ update_reader_status_file (0); /* Update slot status table. */
update_card_removed (slot, 0); /* Clear card_removed flag. */ update_card_removed (vrdr, 0); /* Clear card_removed flag. */
if (!pth_mutex_release (&status_file_update_lock)) if (!pth_mutex_release (&status_file_update_lock))
log_error ("failed to release status_file_update lock\n"); log_error ("failed to release status_file_update lock\n");
/* Do this last, so that the update_card_removed above does its job. */ /* Do this last, so that the update_card_removed above does its job. */
ctrl->reader_slot = -1; ctrl->server_local->vreader_idx = -1;
} }
@ -379,35 +397,38 @@ option_handler (assuan_context_t ctx, const char *key, const char *value)
} }
/* Return the slot of the current reader or open the reader if no /* Return the index of the current reader or open the reader if no
other sessions are using a reader. Note, that we currently support other sessions are using that reader. If it is not possible to
open the reader -1 is returned. Note, that we currently support
only one reader but most of the code (except for this function) only one reader but most of the code (except for this function)
should be able to cope with several readers. */ should be able to cope with several readers. */
static int static int
get_reader_slot (void) get_current_reader (void)
{ {
struct slot_status_s *ss; struct vreader_s *vr;
ss = &slot_table[0]; /* One reader for now. */ /* We only support one reader for now. */
vr = &vreader_table[0];
/* Initialize the item if needed. */ /* Initialize the vreader item if not yet done. */
if (!ss->valid) if (!vr->valid)
{ {
ss->slot = -1; vr->slot = -1;
ss->valid = 1; vr->valid = 1;
} }
/* Try to open the reader. */ /* Try to open the reader. */
if (ss->slot == -1) if (vr->slot == -1)
{ {
int no_service_flag; int no_service_flag;
ss->slot = apdu_open_reader (opt.reader_port, &no_service_flag);
vr->slot = apdu_open_reader (opt.reader_port, &no_service_flag);
/* If we still don't have a slot, we have no readers. /* If we still don't have a slot, we have no readers.
Invalidate for now until a reader is attached. */ Invalidate for now until a reader is attached. */
if(ss->slot == -1) if (vr->slot == -1)
{ {
ss->valid = 0; vr->valid = 0;
} }
if (no_service_flag) if (no_service_flag)
@ -417,8 +438,8 @@ get_reader_slot (void)
} }
} }
/* Return the slot_table index. */ /* Return the vreader index or -1. */
return 0; return vr->valid ? 0 : -1;
} }
@ -427,7 +448,7 @@ static gpg_error_t
open_card (ctrl_t ctrl, const char *apptype) open_card (ctrl_t ctrl, const char *apptype)
{ {
gpg_error_t err; gpg_error_t err;
int slot; int vrdr;
if (reader_disabled) if (reader_disabled)
return gpg_error (GPG_ERR_NOT_OPERATIONAL); return gpg_error (GPG_ERR_NOT_OPERATIONAL);
@ -455,21 +476,25 @@ open_card (ctrl_t ctrl, const char *apptype)
need to check that the client didn't requested a specific need to check that the client didn't requested a specific
application different from the one in use before we continue. */ application different from the one in use before we continue. */
if (ctrl->app_ctx) if (ctrl->app_ctx)
return check_application_conflict (ctrl, apptype); {
return check_application_conflict
(ctrl, vreader_slot (ctrl->server_local->vreader_idx), apptype);
}
/* Setup the slot and select the application. */ /* Setup the vreader and select the application. */
if (ctrl->reader_slot != -1) if (ctrl->server_local->vreader_idx != -1)
slot = ctrl->reader_slot; vrdr = ctrl->server_local->vreader_idx;
else else
slot = get_reader_slot (); vrdr = get_current_reader ();
ctrl->reader_slot = slot; ctrl->server_local->vreader_idx = vrdr;
if (slot == -1) if (vrdr == -1)
err = gpg_error (reader_disabled? GPG_ERR_NOT_OPERATIONAL: GPG_ERR_CARD); err = gpg_error (reader_disabled? GPG_ERR_NOT_OPERATIONAL: GPG_ERR_CARD);
else else
{ {
/* Fixme: We should move the apdu_connect call to /* Fixme: We should move the apdu_connect call to
select_application. */ select_application. */
int sw; int sw;
int slot = vreader_slot (vrdr);
ctrl->server_local->disconnect_allowed = 0; ctrl->server_local->disconnect_allowed = 0;
sw = apdu_connect (slot); sw = apdu_connect (slot);
@ -1615,8 +1640,8 @@ static const char hlp_getinfo[] =
"\n" "\n"
"socket_name - Return the name of the socket.\n" "socket_name - Return the name of the socket.\n"
"\n" "\n"
"status - Return the status of the current slot (in the future, may\n" "status - Return the status of the current reader (in the future, may\n"
"also return the status of all slots). The status is a list of\n" "also return the status of all readers). The status is a list of\n"
"one-character flags. The following flags are currently defined:\n" "one-character flags. The following flags are currently defined:\n"
" 'u' Usable card present. This is the normal state during operation.\n" " 'u' Usable card present. This is the normal state during operation.\n"
" 'r' Card removed. A reset is necessary.\n" " 'r' Card removed. A reset is necessary.\n"
@ -1660,22 +1685,22 @@ cmd_getinfo (assuan_context_t ctx, char *line)
else if (!strcmp (line, "status")) else if (!strcmp (line, "status"))
{ {
ctrl_t ctrl = assuan_get_pointer (ctx); ctrl_t ctrl = assuan_get_pointer (ctx);
int slot = ctrl->reader_slot; int vrdr = ctrl->server_local->vreader_idx;
char flag = 'r'; char flag = 'r';
if (!ctrl->server_local->card_removed && slot != -1) if (!ctrl->server_local->card_removed && vrdr != -1)
{ {
struct slot_status_s *ss; struct vreader_s *vr;
if (!(slot >= 0 && slot < DIM(slot_table))) if (!(vrdr >= 0 && vrdr < DIM(vreader_table)))
BUG (); BUG ();
ss = &slot_table[slot]; vr = &vreader_table[vrdr];
if (!ss->valid) if (!vr->valid)
BUG (); BUG ();
if (ss->any && (ss->status & 1)) if (vr->any && (vr->status & 1))
flag = 'u'; flag = 'u';
} }
rc = assuan_send_data (ctx, &flag, 1); rc = assuan_send_data (ctx, &flag, 1);
@ -1790,6 +1815,7 @@ cmd_apdu (assuan_context_t ctx, char *line)
int handle_more; int handle_more;
const char *s; const char *s;
size_t exlen; size_t exlen;
int slot;
with_atr = has_option (line, "--atr"); with_atr = has_option (line, "--atr");
handle_more = has_option (line, "--more"); handle_more = has_option (line, "--more");
@ -1812,13 +1838,15 @@ cmd_apdu (assuan_context_t ctx, char *line)
if ((rc = open_card (ctrl, NULL))) if ((rc = open_card (ctrl, NULL)))
return rc; return rc;
slot = vreader_slot (ctrl->server_local->vreader_idx);
if (with_atr) if (with_atr)
{ {
unsigned char *atr; unsigned char *atr;
size_t atrlen; size_t atrlen;
char hexbuf[400]; char hexbuf[400];
atr = apdu_get_atr (ctrl->reader_slot, &atrlen); atr = apdu_get_atr (slot, &atrlen);
if (!atr || atrlen > sizeof hexbuf - 2 ) if (!atr || atrlen > sizeof hexbuf - 2 )
{ {
rc = gpg_error (GPG_ERR_INV_CARD); rc = gpg_error (GPG_ERR_INV_CARD);
@ -1840,7 +1868,7 @@ cmd_apdu (assuan_context_t ctx, char *line)
unsigned char *result = NULL; unsigned char *result = NULL;
size_t resultlen; size_t resultlen;
rc = apdu_send_direct (ctrl->reader_slot, exlen, rc = apdu_send_direct (slot, exlen,
apdu, apdulen, handle_more, apdu, apdulen, handle_more,
&result, &resultlen); &result, &resultlen);
if (rc) if (rc)
@ -1990,12 +2018,13 @@ scd_command_handler (ctrl_t ctrl, int fd)
session_list = ctrl->server_local; session_list = ctrl->server_local;
ctrl->server_local->ctrl_backlink = ctrl; ctrl->server_local->ctrl_backlink = ctrl;
ctrl->server_local->assuan_ctx = ctx; ctrl->server_local->assuan_ctx = ctx;
ctrl->server_local->vreader_idx = -1;
/* We open the reader right at startup so that the ticker is able to /* We open the reader right at startup so that the ticker is able to
update the status file. */ update the status file. */
if (ctrl->reader_slot == -1) if (ctrl->server_local->vreader_idx == -1)
{ {
ctrl->reader_slot = get_reader_slot (); ctrl->server_local->vreader_idx = get_current_reader ();
} }
/* Command processing loop. */ /* Command processing loop. */
@ -2197,32 +2226,33 @@ update_reader_status_file (int set_card_removed_flag)
int idx; int idx;
unsigned int status, changed; unsigned int status, changed;
/* Make sure that the reader has been opened. Like get_reader_slot, /* Make sure that a reader has been opened. Like get_current_reader,
this part of the code assumes that there is only one reader. */ this part of the code assumes that there is only one reader. */
if (!slot_table[0].valid) if (!vreader_table[0].valid)
(void)get_reader_slot (); (void)get_current_reader ();
/* Note, that we only try to get the status, because it does not /* Note, that we only try to get the status, because it does not
make sense to wait here for a operation to complete. If we are make sense to wait here for a operation to complete. If we are
busy working with a card, delays in the status file update should busy working with a card, delays in the status file update should
be acceptable. */ be acceptable. */
for (idx=0; idx < DIM(slot_table); idx++) for (idx=0; idx < DIM(vreader_table); idx++)
{ {
struct slot_status_s *ss = slot_table + idx; struct vreader_s *vr = vreader_table + idx;
struct server_local_s *sl; struct server_local_s *sl;
int sw_apdu; int sw_apdu;
if (!ss->valid || ss->slot == -1) if (!vr->valid || vr->slot == -1)
continue; /* Not valid or reader not yet open. */ continue; /* Not valid or reader not yet open. */
sw_apdu = apdu_get_status (ss->slot, 0, &status, &changed); sw_apdu = apdu_get_status (vr->slot, 0, &status, &changed);
if (sw_apdu == SW_HOST_NO_READER) if (sw_apdu == SW_HOST_NO_READER)
{ {
/* Most likely the _reader_ has been unplugged. */ /* Most likely the _reader_ has been unplugged. */
apdu_close_reader(ss->slot); apdu_close_reader (vr->slot);
ss->valid = 0; vr->slot = -1;
vr->valid = 0;
status = 0; status = 0;
changed = ss->changed; changed = vr->changed;
} }
else if (sw_apdu) else if (sw_apdu)
{ {
@ -2230,21 +2260,21 @@ update_reader_status_file (int set_card_removed_flag)
continue; continue;
} }
if (!ss->any || ss->status != status || ss->changed != changed ) if (!vr->any || vr->status != status || vr->changed != changed )
{ {
char *fname; char *fname;
char templ[50]; char templ[50];
FILE *fp; FILE *fp;
log_info ("updating slot %d status: 0x%04X->0x%04X (%u->%u)\n", log_info ("updating reader %d (%d) status: 0x%04X->0x%04X (%u->%u)\n",
ss->slot, ss->status, status, ss->changed, changed); idx, vr->slot, vr->status, status, vr->changed, changed);
ss->status = status; vr->status = status;
ss->changed = changed; vr->changed = changed;
/* FIXME: Should this be IDX instead of ss->slot? This /* FIXME: Should this be IDX instead of vr->slot? This
depends on how client sessions will associate the reader depends on how client sessions will associate the reader
status with their session. */ status with their session. */
snprintf (templ, sizeof templ, "reader_%d.status", ss->slot); snprintf (templ, sizeof templ, "reader_%d.status", vr->slot);
fname = make_filename (opt.homedir, templ, NULL ); fname = make_filename (opt.homedir, templ, NULL );
fp = fopen (fname, "w"); fp = fopen (fname, "w");
if (fp) if (fp)
@ -2272,8 +2302,8 @@ update_reader_status_file (int set_card_removed_flag)
envs[0] = envstr; envs[0] = envstr;
envs[1] = NULL; envs[1] = NULL;
sprintf (numbuf1, "%d", ss->slot); sprintf (numbuf1, "%d", vr->slot);
sprintf (numbuf2, "0x%04X", ss->status); sprintf (numbuf2, "0x%04X", vr->status);
sprintf (numbuf3, "0x%04X", status); sprintf (numbuf3, "0x%04X", status);
args[0] = "--reader-port"; args[0] = "--reader-port";
args[1] = numbuf1; args[1] = numbuf1;
@ -2301,10 +2331,10 @@ update_reader_status_file (int set_card_removed_flag)
/* Set the card removed flag for all current sessions. We /* Set the card removed flag for all current sessions. We
will set this on any card change because a reset or will set this on any card change because a reset or
SERIALNO request must be done in any case. */ SERIALNO request must be done in any case. */
if (ss->any && set_card_removed_flag) if (vr->any && set_card_removed_flag)
update_card_removed (idx, 1); update_card_removed (idx, 1);
ss->any = 1; vr->any = 1;
/* Send a signal to all clients who applied for it. */ /* Send a signal to all clients who applied for it. */
send_client_notifications (); send_client_notifications ();
@ -2320,8 +2350,9 @@ update_reader_status_file (int set_card_removed_flag)
{ {
/* FIXME: Use a real timeout. */ /* FIXME: Use a real timeout. */
/* At least one connection and all allow a disconnect. */ /* At least one connection and all allow a disconnect. */
log_info ("disconnecting card in slot %d\n", ss->slot); log_info ("disconnecting card in reader %d (%d)\n",
apdu_disconnect (ss->slot); idx, vr->slot);
apdu_disconnect (vr->slot);
} }
} }

View File

@ -913,7 +913,7 @@ scd_exit (int rc)
static void static void
scd_init_default_ctrl (ctrl_t ctrl) scd_init_default_ctrl (ctrl_t ctrl)
{ {
ctrl->reader_slot = -1; (void)ctrl;
} }
static void static void

View File

@ -97,9 +97,6 @@ struct server_control_s
/* Local data of the server; used only in command.c. */ /* Local data of the server; used only in command.c. */
struct server_local_s *server_local; struct server_local_s *server_local;
/* Slot of the open reader or -1 if not open. */
int reader_slot;
/* The application context used with this connection or NULL if none /* The application context used with this connection or NULL if none
associated. Note that this is shared with the other connections: associated. Note that this is shared with the other connections:
All connections accessing the same reader are using the same All connections accessing the same reader are using the same