Clean up the provider status logic.
Provider status (done, running, not run) is now attached to the provider-specific data of the client. A reference count of auth instances is kept in the auth_client struct to determine if a client is done or not. This also moves a lot of the logic for manipulating provider data into into the provider.h header for inlining (no point in a function call for these simple accessors).
This commit is contained in:
parent
a68d9a2b61
commit
376ae2e2a7
6 changed files with 89 additions and 70 deletions
|
@ -159,7 +159,7 @@ cancel_providers(struct auth_client *auth)
|
|||
{
|
||||
provider = ptr->data;
|
||||
|
||||
if(provider->cancel && is_provider_on(auth, provider->id))
|
||||
if(provider->cancel && is_provider_running(auth, provider->id))
|
||||
/* Cancel if required */
|
||||
provider->cancel(auth);
|
||||
}
|
||||
|
@ -176,13 +176,12 @@ provider_done(struct auth_client *auth, provider_t id)
|
|||
rb_dlink_node *ptr;
|
||||
struct auth_provider *provider;
|
||||
|
||||
set_provider_off(auth, id);
|
||||
set_provider_done(auth, id);
|
||||
|
||||
if(!auth->providers)
|
||||
if(!auth->refcount)
|
||||
{
|
||||
if(!auth->providers_starting)
|
||||
/* Only do this when there are no providers left */
|
||||
/* Providers aren't being executed and refcount is 0, accept */
|
||||
accept_client(auth, -1);
|
||||
return;
|
||||
}
|
||||
|
@ -191,7 +190,7 @@ provider_done(struct auth_client *auth, provider_t id)
|
|||
{
|
||||
provider = ptr->data;
|
||||
|
||||
if(provider->completed != NULL && is_provider_on(auth, provider->id))
|
||||
if(provider->completed != NULL && is_provider_running(auth, provider->id))
|
||||
/* Notify pending clients who asked for it */
|
||||
provider->completed(auth, id);
|
||||
}
|
||||
|
@ -237,7 +236,7 @@ reject_client(struct auth_client *auth, provider_t id, const char *data, const c
|
|||
*/
|
||||
rb_helper_write(authd_helper, "R %x %c %s %s %s :%s", auth->cid, reject, auth->username, auth->hostname, data, buf);
|
||||
|
||||
set_provider_off(auth, id);
|
||||
set_provider_done(auth, id);
|
||||
cancel_providers(auth);
|
||||
}
|
||||
|
||||
|
@ -247,7 +246,7 @@ accept_client(struct auth_client *auth, provider_t id)
|
|||
{
|
||||
rb_helper_write(authd_helper, "A %x %s %s", auth->cid, auth->username, auth->hostname);
|
||||
|
||||
set_provider_off(auth, id);
|
||||
set_provider_done(auth, id);
|
||||
cancel_providers(auth);
|
||||
}
|
||||
|
||||
|
@ -307,7 +306,7 @@ start_auth(const char *cid, const char *l_ip, const char *l_port, const char *c_
|
|||
auth->providers_starting = false;
|
||||
|
||||
/* If no providers are running, accept the client */
|
||||
if(!auth->providers)
|
||||
if(!auth->refcount)
|
||||
accept_client(auth, -1);
|
||||
}
|
||||
|
||||
|
@ -368,7 +367,7 @@ provider_timeout_event(void *notused __unused)
|
|||
struct auth_provider *provider = ptr->data;
|
||||
const time_t timeout = get_provider_timeout(auth, provider->id);
|
||||
|
||||
if(is_provider_on(auth, provider->id) && provider->timeout != NULL &&
|
||||
if(is_provider_running(auth, provider->id) && provider->timeout != NULL &&
|
||||
timeout > 0 && timeout < curtime)
|
||||
{
|
||||
provider->timeout(auth);
|
||||
|
@ -376,38 +375,3 @@ provider_timeout_event(void *notused __unused)
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
void *
|
||||
get_provider_data(struct auth_client *auth, uint32_t id)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
return auth->data[id].data;
|
||||
}
|
||||
|
||||
void
|
||||
set_provider_data(struct auth_client *auth, uint32_t id, void *data)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
auth->data[id].data = data;
|
||||
}
|
||||
|
||||
void
|
||||
set_provider_timeout_relative(struct auth_client *auth, uint32_t id, time_t timeout)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
auth->data[id].timeout = timeout + rb_current_time();
|
||||
}
|
||||
|
||||
void
|
||||
set_provider_timeout_absolute(struct auth_client *auth, uint32_t id, time_t timeout)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
auth->data[id].timeout = timeout;
|
||||
}
|
||||
|
||||
time_t
|
||||
get_provider_timeout(struct auth_client *auth, uint32_t id)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
return auth->data[id].timeout;
|
||||
}
|
||||
|
|
|
@ -36,10 +36,18 @@ typedef enum
|
|||
PROVIDER_OPM,
|
||||
} provider_t;
|
||||
|
||||
typedef enum
|
||||
{
|
||||
PROVIDER_STATUS_NOTRUN = 0,
|
||||
PROVIDER_STATUS_RUNNING,
|
||||
PROVIDER_STATUS_DONE,
|
||||
} provider_status_t;
|
||||
|
||||
struct auth_client_data
|
||||
{
|
||||
time_t timeout; /* Provider timeout */
|
||||
void *data; /* Proivder data */
|
||||
time_t timeout; /* Provider timeout */
|
||||
void *data; /* Provider data */
|
||||
provider_status_t status; /* Provider status */
|
||||
};
|
||||
|
||||
struct auth_client
|
||||
|
@ -57,10 +65,8 @@ struct auth_client
|
|||
char hostname[HOSTLEN + 1]; /* Used for DNS lookup */
|
||||
char username[USERLEN + 1]; /* Used for ident lookup */
|
||||
|
||||
uint32_t providers; /* Providers at work,
|
||||
* none left when set to 0 */
|
||||
uint32_t providers_done; /* Providers completed */
|
||||
bool providers_starting; /* Providers are still warming up */
|
||||
unsigned int refcount; /* Held references */
|
||||
|
||||
struct auth_client_data *data; /* Provider-specific data */
|
||||
};
|
||||
|
@ -120,44 +126,93 @@ void reject_client(struct auth_client *auth, provider_t id, const char *data, co
|
|||
void handle_new_connection(int parc, char *parv[]);
|
||||
void handle_cancel_connection(int parc, char *parv[]);
|
||||
|
||||
void *get_provider_data(struct auth_client *auth, uint32_t id);
|
||||
void set_provider_data(struct auth_client *auth, uint32_t id, void *data);
|
||||
void set_provider_timeout_relative(struct auth_client *auth, uint32_t id, time_t timeout);
|
||||
void set_provider_timeout_absolute(struct auth_client *auth, uint32_t id, time_t timeout);
|
||||
time_t get_provider_timeout(struct auth_client *auth, uint32_t id);
|
||||
|
||||
/* Provider is operating on this auth_client (set this if you have async work to do) */
|
||||
static inline void
|
||||
set_provider_on(struct auth_client *auth, provider_t provider)
|
||||
/* Get a provider's raw status */
|
||||
static inline provider_status_t
|
||||
get_provider_status(struct auth_client *auth, provider_t provider)
|
||||
{
|
||||
auth->providers |= (1 << provider);
|
||||
return auth->data[provider].status;
|
||||
}
|
||||
|
||||
/* Provider is no longer operating on this auth client (you should use provider_done) */
|
||||
/* Set a provider's raw status */
|
||||
static inline void
|
||||
set_provider_off(struct auth_client *auth, provider_t provider)
|
||||
set_provider_status(struct auth_client *auth, provider_t provider, provider_status_t status)
|
||||
{
|
||||
auth->providers &= ~(1 << provider);
|
||||
auth->data[provider].status = status;
|
||||
}
|
||||
|
||||
/* Set the provider to done (you should use provider_done) */
|
||||
/* Set the provider as running
|
||||
* If you're doing asynchronous work call this */
|
||||
static inline void
|
||||
set_provider_running(struct auth_client *auth, provider_t provider)
|
||||
{
|
||||
auth->refcount++;
|
||||
set_provider_status(auth, provider, PROVIDER_STATUS_RUNNING);
|
||||
}
|
||||
|
||||
/* Provider is no longer operating on this auth client
|
||||
* You should use provider_done and not this */
|
||||
static inline void
|
||||
set_provider_done(struct auth_client *auth, provider_t provider)
|
||||
{
|
||||
auth->providers_done |= (1 << provider);
|
||||
auth->refcount--;
|
||||
set_provider_status(auth, provider, PROVIDER_STATUS_DONE);
|
||||
}
|
||||
|
||||
/* Check if provider is operating on this auth client */
|
||||
static inline bool
|
||||
is_provider_on(struct auth_client *auth, provider_t provider)
|
||||
is_provider_running(struct auth_client *auth, provider_t provider)
|
||||
{
|
||||
return auth->providers & (1 << provider);
|
||||
return get_provider_status(auth, provider) == PROVIDER_STATUS_RUNNING;
|
||||
}
|
||||
|
||||
/* Check if provider has finished on this client */
|
||||
static inline bool
|
||||
is_provider_done(struct auth_client *auth, provider_t provider)
|
||||
{
|
||||
return auth->providers_done & (1 << provider);
|
||||
return get_provider_status(auth, provider) == PROVIDER_STATUS_DONE;
|
||||
}
|
||||
|
||||
/* Get provider auth client data */
|
||||
static inline void *
|
||||
get_provider_data(struct auth_client *auth, uint32_t id)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
return auth->data[id].data;
|
||||
}
|
||||
|
||||
/* Set provider auth client data */
|
||||
static inline void
|
||||
set_provider_data(struct auth_client *auth, uint32_t id, void *data)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
auth->data[id].data = data;
|
||||
}
|
||||
|
||||
/* Set timeout relative to current time on provider
|
||||
* When the timeout lapses, the provider's timeout call will execute */
|
||||
static inline void
|
||||
set_provider_timeout_relative(struct auth_client *auth, uint32_t id, time_t timeout)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
auth->data[id].timeout = timeout + rb_current_time();
|
||||
}
|
||||
|
||||
/* Set timeout value in absolute time (Unix timestamp)
|
||||
* When the timeout lapses, the provider's timeout call will execute */
|
||||
static inline void
|
||||
set_provider_timeout_absolute(struct auth_client *auth, uint32_t id, time_t timeout)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
auth->data[id].timeout = timeout;
|
||||
}
|
||||
|
||||
/* Get the timeout value for the provider */
|
||||
static inline time_t
|
||||
get_provider_timeout(struct auth_client *auth, uint32_t id)
|
||||
{
|
||||
lrb_assert(id < rb_dlink_list_length(&auth_providers));
|
||||
return auth->data[id].timeout;
|
||||
}
|
||||
|
||||
#endif /* __CHARYBDIS_AUTHD_PROVIDER_H__ */
|
||||
|
|
|
@ -356,7 +356,7 @@ blacklists_start(struct auth_client *auth)
|
|||
/* This probably can't happen but let's handle this case anyway */
|
||||
lookup_all_blacklists(auth);
|
||||
|
||||
set_provider_on(auth, PROVIDER_BLACKLIST);
|
||||
set_provider_running(auth, PROVIDER_BLACKLIST);
|
||||
return true;
|
||||
}
|
||||
|
||||
|
|
|
@ -324,7 +324,7 @@ ident_start(struct auth_client *auth)
|
|||
GET_SS_LEN(&l_addr), ident_connected,
|
||||
auth, ident_timeout);
|
||||
|
||||
set_provider_on(auth, PROVIDER_IDENT);
|
||||
set_provider_running(auth, PROVIDER_IDENT);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
|
|
@ -669,7 +669,7 @@ opm_start(struct auth_client *auth)
|
|||
/* This probably can't happen but let's handle this case anyway */
|
||||
opm_scan(auth);
|
||||
|
||||
set_provider_on(auth, PROVIDER_BLACKLIST);
|
||||
set_provider_running(auth, PROVIDER_BLACKLIST);
|
||||
return true;
|
||||
}
|
||||
|
||||
|
|
|
@ -131,7 +131,7 @@ rdns_start(struct auth_client *auth)
|
|||
query->query = lookup_hostname(auth->c_ip, dns_answer_callback, auth);
|
||||
|
||||
notice_client(auth->cid, messages[REPORT_LOOKUP]);
|
||||
set_provider_on(auth, PROVIDER_RDNS);
|
||||
set_provider_running(auth, PROVIDER_RDNS);
|
||||
return true;
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in a new issue