diff options
| author | Vikas Gorur <vikas@gluster.com> | 2009-09-28 20:58:57 +0000 | 
|---|---|---|
| committer | Anand V. Avati <avati@dev.gluster.com> | 2009-09-28 19:36:12 -0700 | 
| commit | 359c46118b756e1a5191ec6e902709258aad903b (patch) | |
| tree | 80de05d5da31ea5319f3e1daab5fde062d72b171 | |
| parent | f09558e2bf5b39284c6a2b750d791d38854ffcfc (diff) | |
cluster/afr: dir-write: Fix inode number handling.
create, mkdir, symlink, mknod: Prefer to return itransform'd inode number
from the first_up_child. If not, fall back on any other child that returned
succcess.
link, rename: Return the same inode number that
was passed as part of loc_t.
Also adds a new member to afr_local_t, local->first_up_child
which is initialized at the start of the transaction. This
fixes the race where a subvolume might go down during the transaction
and thus have the first_up_child change.
Signed-off-by: Anand V. Avati <avati@dev.gluster.com>
BUG: 285 ("first up child" can change during a transaction)
URL: http://bugs.gluster.com/cgi-bin/bugzilla3/show_bug.cgi?id=285
| -rw-r--r-- | xlators/cluster/afr/src/afr-dir-write.c | 64 | ||||
| -rw-r--r-- | xlators/cluster/afr/src/afr.h | 1 | 
2 files changed, 41 insertions, 24 deletions
| diff --git a/xlators/cluster/afr/src/afr-dir-write.c b/xlators/cluster/afr/src/afr-dir-write.c index 7be32ce4c..51629234f 100644 --- a/xlators/cluster/afr/src/afr-dir-write.c +++ b/xlators/cluster/afr/src/afr-dir-write.c @@ -124,7 +124,6 @@ afr_create_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,  	int call_count = -1;  	int child_index = -1; -        int first_up_child = 0;  	local = frame->local;  	priv  = this->private; @@ -153,6 +152,11 @@ afr_create_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,                          if (local->success_count == 0) {  				local->cont.create.buf        = *buf; +                                local->cont.create.ino = +                                        afr_itransform (buf->st_ino, +                                                        priv->child_count, +                                                        child_index); +                                  if (priv->read_child >= 0) {                                          afr_set_read_child (this, inode,                                                               priv->read_child); @@ -162,14 +166,12 @@ afr_create_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,                                  }  			} -                        first_up_child = afr_first_up_child (priv); - -                        if (child_index == first_up_child) { +                        if (child_index == local->first_up_child) {                                  local->cont.create.ino =                                          afr_itransform (buf->st_ino,                                                          priv->child_count, -                                                        first_up_child); -                } +                                                        local->first_up_child); +                        }                          if (child_index == local->read_child_index) {                                  local->cont.create.read_child_buf = *buf; @@ -352,6 +354,8 @@ afr_mknod_unwind (call_frame_t *frame, xlator_t *this)                          unwind_buf = &local->cont.mknod.buf;                  } +                unwind_buf->st_ino = local->cont.mknod.ino; +  		AFR_STACK_UNWIND (main_frame, local->op_ret, local->op_errno,  				  local->cont.mknod.inode,  				  unwind_buf); @@ -387,7 +391,7 @@ afr_mknod_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,  			if (local->success_count == 0){  				local->cont.mknod.buf   = *buf; -				local->cont.mknod.buf.st_ino =  +				local->cont.mknod.ino   =   					afr_itransform (buf->st_ino,  							priv->child_count,  							child_index); @@ -401,6 +405,13 @@ afr_mknod_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,                                  }  			} +                        if (child_index == local->first_up_child) { +                                local->cont.mknod.ino = +                                        afr_itransform (buf->st_ino, +                                                        priv->child_count, +                                                        local->first_up_child); +                        } +                          if (child_index == local->read_child_index) {                                  local->cont.mknod.read_child_buf = *buf;                          } @@ -599,7 +610,6 @@ afr_mkdir_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,  	int call_count = -1;  	int child_index = -1; -        int first_up_child = 0;  	local = frame->local;  	priv = this->private; @@ -617,6 +627,11 @@ afr_mkdir_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,  			if (local->success_count == 0) {  				local->cont.mkdir.buf   = *buf; +                                local->cont.mkdir.ino = +                                        afr_itransform (buf->st_ino, +                                                        priv->child_count, +                                                        child_index); +                                  if (priv->read_child >= 0) {                                          afr_set_read_child (this, inode,                                                              priv->read_child); @@ -626,13 +641,11 @@ afr_mkdir_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,                                  }  			} -                        first_up_child = afr_first_up_child (priv); - -                        if (child_index == first_up_child) { +                        if (child_index == local->first_up_child) {                                  local->cont.mkdir.ino =                                          afr_itransform (buf->st_ino,                                                          priv->child_count, -                                                        first_up_child); +                                                        local->first_up_child);                          }                          if (child_index == local->read_child_index) { @@ -850,10 +863,7 @@ afr_link_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,  			if (local->success_count == 0) {  				local->cont.link.buf        = *buf; -				local->cont.link.buf.st_ino =  -					afr_itransform (buf->st_ino, priv->child_count, -							child_index); -                                 +                                  if (priv->read_child >= 0) {                                          afr_set_read_child (this, inode,                                                              priv->read_child); @@ -862,7 +872,7 @@ afr_link_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,                                                              local->read_child_index);                                  }  			} -                         +                          if (child_index == local->read_child_index) {                                  local->cont.link.read_child_buf = *buf;                          } @@ -1041,6 +1051,8 @@ afr_symlink_unwind (call_frame_t *frame, xlator_t *this)                          unwind_buf = &local->cont.symlink.buf;                  } +                unwind_buf->st_ino = local->cont.symlink.ino; +  		AFR_STACK_UNWIND (main_frame, local->op_ret, local->op_errno,  				  local->cont.symlink.inode,  				  unwind_buf); @@ -1076,7 +1088,7 @@ afr_symlink_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,  			if (local->success_count == 0) {  				local->cont.symlink.buf        = *buf; -				local->cont.symlink.buf.st_ino =  +				local->cont.symlink.ino =   					afr_itransform (buf->st_ino, priv->child_count,  							child_index); @@ -1089,6 +1101,13 @@ afr_symlink_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,                                  }  			} +                        if (child_index == local->first_up_child) { +                                local->cont.symlink.ino = +                                        afr_itransform (buf->st_ino, +                                                        priv->child_count, +                                                        local->first_up_child); +                        } +                          if (child_index == local->read_child_index) {                                  local->cont.symlink.read_child_buf = *buf;                          } @@ -1212,7 +1231,6 @@ afr_symlink (call_frame_t *frame, xlator_t *this,          }          UNLOCK (&priv->read_child_lock); -	local->cont.symlink.ino      = loc->inode->ino;  	local->cont.symlink.linkpath = strdup (linkpath);  	local->transaction.fop    = afr_symlink_wind; @@ -1261,14 +1279,14 @@ afr_rename_unwind (call_frame_t *frame, xlator_t *this)  	UNLOCK (&frame->lock);  	if (main_frame) { -		local->cont.rename.buf.st_ino = local->cont.rename.ino; -                  if (local->cont.rename.read_child_buf.st_ino) {                          unwind_buf = &local->cont.rename.read_child_buf;                  } else {                          unwind_buf = &local->cont.rename.buf;                  } +		unwind_buf->st_ino = local->cont.rename.ino; +  		AFR_STACK_UNWIND (main_frame, local->op_ret, local->op_errno,   				  unwind_buf);  	} @@ -1303,10 +1321,8 @@ afr_rename_wind_cbk (call_frame_t *frame, void *cookie, xlator_t *this,                                  if (buf) {                                          local->cont.rename.buf = *buf; -                                        local->cont.rename.buf.st_ino =  -                                                afr_itransform (buf->st_ino, priv->child_count, -                                                                child_index);                                  } +                                  local->success_count++;                          } diff --git a/xlators/cluster/afr/src/afr.h b/xlators/cluster/afr/src/afr.h index d9a3435c3..87e210b23 100644 --- a/xlators/cluster/afr/src/afr.h +++ b/xlators/cluster/afr/src/afr.h @@ -146,6 +146,7 @@ typedef struct _afr_local {  	unsigned int read_child_index;          unsigned char read_child_returned; +        unsigned int first_up_child;          pid_t saved_pid; | 
