diff options
| author | Krishnan Parthasarathi <kparthas@redhat.com> | 2013-04-15 15:56:57 +0530 | 
|---|---|---|
| committer | Vijay Bellur <vbellur@redhat.com> | 2013-04-17 05:48:50 -0700 | 
| commit | 92729add67e2e7b8c7589c2dfab0bde071a7faf2 (patch) | |
| tree | 4e8c2d48de34938a71abd320402e98e1588aee30 /xlators/mgmt/glusterd/src/glusterd-handshake.c | |
| parent | 47c118e22d9d6fb6662fe96841ed4fe3089739b5 (diff) | |
glusterd: big lock - a coarse-grained locking to prevent racesv3.4.0alpha3
There are primarily three lists that are part of glusterd process,
that are concurrently accessed. Namely, priv->volumes, priv->peers
and volinfo->bricks_list.
Big-lock approach
-----------------
WHAT IS IT?
Big lock is a coarse-grained lock which protects all three
lists, mentioned above, from racy access.
HOW DOES IT WORK?
At any given point in time, glusterd's thread(s) are in execution
_iff_ there is a preceding, inbound network event. Of course, the
sigwaiter thread and timer thread are exceptions.
A network event is an external trigger to glusterd, via the epoll
thread, in the form of POLLIN and POLLERR.
As long as we take the big-lock at all such entry points and yield
it when we are done, we are guaranteed that all the network events,
accessing the global lists, are serialised.
This amounts to holding the big lock at
- all the handlers of all the actors in glusterd. (POLLIN)
- all the cbks in glusterd. (POLLIN)
- rpc_notify (DISCONNECT event), if we access/modify
  one of the three lists. (POLLERR)
In the case of synctask'ized volume operations, we must remember that,
if we held the big lock for the entire duration of the handler,
we may block other non-synctask rpc actors from executing.
For eg, volume-start would block in PMAP SIGNIN, if done incorrectly.
To prevent this, we need to yield the big lock, when we yield the
synctask, and reacquire on waking up of the synctask.
BUG: 948686
Change-Id: I429832f1fed67bcac0813403d58346558a403ce9
Signed-off-by: Krishnan Parthasarathi <kparthas@redhat.com>
Reviewed-on: http://review.gluster.org/4835
Tested-by: Gluster Build System <jenkins@build.gluster.com>
Reviewed-by: Vijay Bellur <vbellur@redhat.com>
Diffstat (limited to 'xlators/mgmt/glusterd/src/glusterd-handshake.c')
| -rw-r--r-- | xlators/mgmt/glusterd/src/glusterd-handshake.c | 63 | 
1 files changed, 56 insertions, 7 deletions
| diff --git a/xlators/mgmt/glusterd/src/glusterd-handshake.c b/xlators/mgmt/glusterd/src/glusterd-handshake.c index 8b9647026..c41a94598 100644 --- a/xlators/mgmt/glusterd/src/glusterd-handshake.c +++ b/xlators/mgmt/glusterd/src/glusterd-handshake.c @@ -109,7 +109,7 @@ out:  }  int -server_getspec (rpcsvc_request_t *req) +__server_getspec (rpcsvc_request_t *req)  {          int32_t               ret                    = -1;          int32_t               op_errno               = 0; @@ -289,8 +289,14 @@ fail:          return 0;  } +int +server_getspec (rpcsvc_request_t *req) +{ +        return glusterd_big_locked_handler (req, __server_getspec); +} +  int32_t -server_event_notify (rpcsvc_request_t *req) +__server_event_notify (rpcsvc_request_t *req)  {          int32_t                 ret             = -1;          int32_t                 op_errno        =  0; @@ -350,6 +356,12 @@ fail:          return 0;  } +int32_t +server_event_notify (rpcsvc_request_t *req) +{ +        return glusterd_big_locked_handler (req, __server_event_notify); +} +  int  gd_validate_cluster_op_version (xlator_t *this, int cluster_op_version,                                  char *peerid) @@ -383,7 +395,7 @@ out:  }  int -glusterd_mgmt_hndsk_versions (rpcsvc_request_t *req) +__glusterd_mgmt_hndsk_versions (rpcsvc_request_t *req)  {          dict_t            *dict            = NULL;          xlator_t          *this            = NULL; @@ -459,7 +471,14 @@ out:  }  int -glusterd_mgmt_hndsk_versions_ack (rpcsvc_request_t *req) +glusterd_mgmt_hndsk_versions (rpcsvc_request_t *req) +{ +        return glusterd_big_locked_handler (req, +                                            __glusterd_mgmt_hndsk_versions); +} + +int +__glusterd_mgmt_hndsk_versions_ack (rpcsvc_request_t *req)  {          dict_t            *clnt_dict       = NULL;          xlator_t          *this            = NULL; @@ -527,6 +546,12 @@ out:          return ret;  } +int +glusterd_mgmt_hndsk_versions_ack (rpcsvc_request_t *req) +{ +        return glusterd_big_locked_handler (req, +                                            __glusterd_mgmt_hndsk_versions_ack); +}  rpcsvc_actor_t gluster_handshake_actors[] = {          [GF_HNDSK_NULL]         = {"NULL", GF_HNDSK_NULL, NULL, NULL, 0}, @@ -699,7 +724,7 @@ out:  }  int -glusterd_mgmt_hndsk_version_ack_cbk (struct rpc_req *req, struct iovec *iov, +__glusterd_mgmt_hndsk_version_ack_cbk (struct rpc_req *req, struct iovec *iov,                                       int count, void *myframe)  {          int                  ret      = -1; @@ -778,7 +803,15 @@ out:  }  int -glusterd_mgmt_hndsk_version_cbk (struct rpc_req *req, struct iovec *iov, +glusterd_mgmt_hndsk_version_ack_cbk (struct rpc_req *req, struct iovec *iov, +                                     int count, void *myframe) +{ +        return glusterd_big_locked_cbk (req, iov, count, myframe, +                                        __glusterd_mgmt_hndsk_version_ack_cbk); +} + +int +__glusterd_mgmt_hndsk_version_cbk (struct rpc_req *req, struct iovec *iov,                                   int count, void *myframe)  {          int                  ret       = -1; @@ -883,6 +916,14 @@ out:  }  int +glusterd_mgmt_hndsk_version_cbk (struct rpc_req *req, struct iovec *iov, +                                     int count, void *myframe) +{ +        return glusterd_big_locked_cbk (req, iov, count, myframe, +                                        __glusterd_mgmt_hndsk_version_cbk); +} + +int  glusterd_mgmt_handshake (xlator_t *this, glusterd_peerctx_t *peerctx)  {          call_frame_t        *frame    = NULL; @@ -985,7 +1026,7 @@ out:  }  int -glusterd_peer_dump_version_cbk (struct rpc_req *req, struct iovec *iov, +__glusterd_peer_dump_version_cbk (struct rpc_req *req, struct iovec *iov,                                  int count, void *myframe)  {          int                  ret      = -1; @@ -1092,6 +1133,14 @@ out:  int +glusterd_peer_dump_version_cbk (struct rpc_req *req, struct iovec *iov, +                                int count, void *myframe) +{ +        return glusterd_big_locked_cbk (req, iov, count, myframe, +                                        __glusterd_peer_dump_version_cbk); +} + +int  glusterd_peer_dump_version (xlator_t *this, struct rpc_clnt *rpc,                              glusterd_peerctx_t *peerctx)  { | 
