mirror of
https://github.com/fluencelabs/redis
synced 2025-04-03 00:01:04 +00:00
Refactoring: unlinkClient() added to lower freeClient() complexity.
This commit is contained in:
parent
fdb3be939e
commit
1e7153831d
@ -685,12 +685,50 @@ void disconnectSlaves(void) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void freeClient(client *c) {
|
/* Remove the specified client from global lists where the client could
|
||||||
|
* be referenced, not including the Pub/Sub channels.
|
||||||
|
* This is used by freeClient() and replicationCacheMaster(). */
|
||||||
|
void unlinkClient(client *c) {
|
||||||
listNode *ln;
|
listNode *ln;
|
||||||
|
|
||||||
/* If this is marked as current client unset it */
|
/* If this is marked as current client unset it. */
|
||||||
if (server.current_client == c) server.current_client = NULL;
|
if (server.current_client == c) server.current_client = NULL;
|
||||||
|
|
||||||
|
/* Certain operations must be done only if the client has an active socket.
|
||||||
|
* If the client was already unlinked or if it's a "fake client" the
|
||||||
|
* fd is already set to -1. */
|
||||||
|
if (c->fd != -1) {
|
||||||
|
/* Remove from the list of active clients. */
|
||||||
|
ln = listSearchKey(server.clients,c);
|
||||||
|
serverAssert(ln != NULL);
|
||||||
|
listDelNode(server.clients,ln);
|
||||||
|
|
||||||
|
/* Unregister async I/O handlers and close the socket. */
|
||||||
|
aeDeleteFileEvent(server.el,c->fd,AE_READABLE);
|
||||||
|
aeDeleteFileEvent(server.el,c->fd,AE_WRITABLE);
|
||||||
|
close(c->fd);
|
||||||
|
c->fd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove from the list of pending writes if needed. */
|
||||||
|
if (c->flags & CLIENT_PENDING_WRITE) {
|
||||||
|
ln = listSearchKey(server.clients_pending_write,c);
|
||||||
|
serverAssert(ln != NULL);
|
||||||
|
listDelNode(server.clients_pending_write,ln);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When client was just unblocked because of a blocking operation,
|
||||||
|
* remove it from the list of unblocked clients. */
|
||||||
|
if (c->flags & CLIENT_UNBLOCKED) {
|
||||||
|
ln = listSearchKey(server.unblocked_clients,c);
|
||||||
|
serverAssert(ln != NULL);
|
||||||
|
listDelNode(server.unblocked_clients,ln);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void freeClient(client *c) {
|
||||||
|
listNode *ln;
|
||||||
|
|
||||||
/* If it is our master that's beging disconnected we should make sure
|
/* If it is our master that's beging disconnected we should make sure
|
||||||
* to cache the state to try a partial resynchronization later.
|
* to cache the state to try a partial resynchronization later.
|
||||||
*
|
*
|
||||||
@ -732,36 +770,14 @@ void freeClient(client *c) {
|
|||||||
dictRelease(c->pubsub_channels);
|
dictRelease(c->pubsub_channels);
|
||||||
listRelease(c->pubsub_patterns);
|
listRelease(c->pubsub_patterns);
|
||||||
|
|
||||||
/* Close socket, unregister events, and remove list of replies and
|
/* Free data structures. */
|
||||||
* accumulated arguments. */
|
|
||||||
if (c->fd != -1) {
|
|
||||||
aeDeleteFileEvent(server.el,c->fd,AE_READABLE);
|
|
||||||
aeDeleteFileEvent(server.el,c->fd,AE_WRITABLE);
|
|
||||||
close(c->fd);
|
|
||||||
}
|
|
||||||
listRelease(c->reply);
|
listRelease(c->reply);
|
||||||
freeClientArgv(c);
|
freeClientArgv(c);
|
||||||
|
|
||||||
/* Remove from the list of clients */
|
/* Unlink the client: this will close the socket, remove the I/O
|
||||||
if (c->fd != -1) {
|
* handlers, and remove references of the client from different
|
||||||
ln = listSearchKey(server.clients,c);
|
* places where active clients may be referenced. */
|
||||||
serverAssert(ln != NULL);
|
unlinkClient(c);
|
||||||
listDelNode(server.clients,ln);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Remove from the list of pending writes if needed. */
|
|
||||||
if (c->flags & CLIENT_PENDING_WRITE) {
|
|
||||||
ln = listSearchKey(server.clients_pending_write,c);
|
|
||||||
listDelNode(server.clients_pending_write,ln);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* When client was just unblocked because of a blocking operation,
|
|
||||||
* remove it from the list of unblocked clients. */
|
|
||||||
if (c->flags & CLIENT_UNBLOCKED) {
|
|
||||||
ln = listSearchKey(server.unblocked_clients,c);
|
|
||||||
serverAssert(ln != NULL);
|
|
||||||
listDelNode(server.unblocked_clients,ln);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Master/slave cleanup Case 1:
|
/* Master/slave cleanup Case 1:
|
||||||
* we lost the connection with a slave. */
|
* we lost the connection with a slave. */
|
||||||
|
@ -1846,36 +1846,16 @@ void replicationSendAck(void) {
|
|||||||
* handshake in order to reactivate the cached master.
|
* handshake in order to reactivate the cached master.
|
||||||
*/
|
*/
|
||||||
void replicationCacheMaster(client *c) {
|
void replicationCacheMaster(client *c) {
|
||||||
listNode *ln;
|
|
||||||
|
|
||||||
serverAssert(server.master != NULL && server.cached_master == NULL);
|
serverAssert(server.master != NULL && server.cached_master == NULL);
|
||||||
serverLog(LL_NOTICE,"Caching the disconnected master state.");
|
serverLog(LL_NOTICE,"Caching the disconnected master state.");
|
||||||
|
|
||||||
/* Remove from the list of clients, we don't want this client to be
|
/* Unlink the client from the server structures. */
|
||||||
* listed by CLIENT LIST or processed in any way by batch operations. */
|
unlinkClient(c);
|
||||||
ln = listSearchKey(server.clients,c);
|
|
||||||
serverAssert(ln != NULL);
|
|
||||||
listDelNode(server.clients,ln);
|
|
||||||
|
|
||||||
/* Remove from the list of clients with pending writes as well. */
|
|
||||||
if (c->flags & CLIENT_PENDING_WRITE) {
|
|
||||||
ln = listSearchKey(server.clients_pending_write,c);
|
|
||||||
if (ln) listDelNode(server.clients_pending_write,ln);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Save the master. Server.master will be set to null later by
|
/* Save the master. Server.master will be set to null later by
|
||||||
* replicationHandleMasterDisconnection(). */
|
* replicationHandleMasterDisconnection(). */
|
||||||
server.cached_master = server.master;
|
server.cached_master = server.master;
|
||||||
|
|
||||||
/* Remove the event handlers and close the socket. We'll later reuse
|
|
||||||
* the socket of the new connection with the master during PSYNC. */
|
|
||||||
aeDeleteFileEvent(server.el,c->fd,AE_READABLE);
|
|
||||||
aeDeleteFileEvent(server.el,c->fd,AE_WRITABLE);
|
|
||||||
close(c->fd);
|
|
||||||
|
|
||||||
/* Set fd to -1 so that we can safely call freeClient(c) later. */
|
|
||||||
c->fd = -1;
|
|
||||||
|
|
||||||
/* Invalidate the Peer ID cache. */
|
/* Invalidate the Peer ID cache. */
|
||||||
if (c->peerid) {
|
if (c->peerid) {
|
||||||
sdsfree(c->peerid);
|
sdsfree(c->peerid);
|
||||||
|
@ -1112,6 +1112,7 @@ int clientsArePaused(void);
|
|||||||
int processEventsWhileBlocked(void);
|
int processEventsWhileBlocked(void);
|
||||||
void handleClientsWithPendingWrites(void);
|
void handleClientsWithPendingWrites(void);
|
||||||
int clientHasPendingReplies(client *c);
|
int clientHasPendingReplies(client *c);
|
||||||
|
void unlinkClient(client *c);
|
||||||
|
|
||||||
#ifdef __GNUC__
|
#ifdef __GNUC__
|
||||||
void addReplyErrorFormat(client *c, const char *fmt, ...)
|
void addReplyErrorFormat(client *c, const char *fmt, ...)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user