diff options
Diffstat (limited to 'xlators')
| -rw-r--r-- | xlators/cluster/afr/src/afr-common.c | 241 | ||||
| -rw-r--r-- | xlators/cluster/afr/src/afr.c | 1 | ||||
| -rw-r--r-- | xlators/cluster/afr/src/afr.h | 8 | ||||
| -rw-r--r-- | xlators/cluster/dht/src/dht-common.h | 2 | ||||
| -rw-r--r-- | xlators/cluster/dht/src/dht-inode-write.c | 136 | ||||
| -rw-r--r-- | xlators/cluster/dht/src/dht.c | 1 | ||||
| -rw-r--r-- | xlators/cluster/stripe/src/stripe.c | 167 | ||||
| -rw-r--r-- | xlators/debug/io-stats/src/io-stats.c | 27 | ||||
| -rw-r--r-- | xlators/features/marker/src/marker.c | 70 | ||||
| -rw-r--r-- | xlators/mount/fuse/src/fuse-bridge.c | 18 | ||||
| -rw-r--r-- | xlators/performance/io-cache/src/io-cache.c | 28 | ||||
| -rw-r--r-- | xlators/performance/io-threads/src/io-threads.c | 52 | ||||
| -rw-r--r-- | xlators/performance/md-cache/src/md-cache.c | 40 | ||||
| -rw-r--r-- | xlators/performance/open-behind/src/open-behind.c | 20 | ||||
| -rw-r--r-- | xlators/performance/read-ahead/src/read-ahead.c | 50 | ||||
| -rw-r--r-- | xlators/protocol/client/src/client-rpc-fops.c | 101 | ||||
| -rw-r--r-- | xlators/protocol/client/src/client.c | 35 | ||||
| -rw-r--r-- | xlators/protocol/server/src/server-rpc-fops.c | 122 | ||||
| -rw-r--r-- | xlators/storage/posix/src/posix.c | 90 | 
19 files changed, 1171 insertions, 38 deletions
diff --git a/xlators/cluster/afr/src/afr-common.c b/xlators/cluster/afr/src/afr-common.c index 830ecd99386..20513d380ba 100644 --- a/xlators/cluster/afr/src/afr-common.c +++ b/xlators/cluster/afr/src/afr-common.c @@ -3065,6 +3065,247 @@ out:  /* }}} */ +/* {{{ discard */ + +static int +afr_discard_unwind (call_frame_t *frame, xlator_t *this) +{ +        afr_local_t *   local = NULL; +        call_frame_t   *main_frame = NULL; + +        local = frame->local; + +        LOCK (&frame->lock); +        { +                if (local->transaction.main_frame) +                        main_frame = local->transaction.main_frame; +                local->transaction.main_frame = NULL; +        } +        UNLOCK (&frame->lock); + +        if (main_frame) { +                AFR_STACK_UNWIND (discard, main_frame, local->op_ret, +                                  local->op_errno, +                                  &local->cont.discard.prebuf, +                                  &local->cont.discard.postbuf, +                                  NULL); +        } +        return 0; +} + +static int +afr_discard_wind_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +                     int32_t op_ret, int32_t op_errno, struct iatt *prebuf, +                     struct iatt *postbuf, dict_t *xdata) +{ +        afr_local_t *   local = NULL; +        afr_private_t * priv  = NULL; +        int child_index = (long) cookie; +        int call_count  = -1; +        int need_unwind = 0; +        int read_child  = 0; + +        local = frame->local; +        priv  = this->private; + +        read_child = afr_inode_get_read_ctx (this, local->fd->inode, NULL); + +        LOCK (&frame->lock); +        { +                if (child_index == read_child) { +                        local->read_child_returned = _gf_true; +                } + +                if (afr_fop_failed (op_ret, op_errno)) +                        afr_transaction_fop_failed (frame, this, child_index); + +                if (op_ret != -1) { +                        if (local->success_count == 0) { +                                local->op_ret = op_ret; +                                local->cont.discard.prebuf  = *prebuf; +                                local->cont.discard.postbuf = *postbuf; +                        } + +                        if (child_index == read_child) { +                                local->cont.discard.prebuf  = *prebuf; +                                local->cont.discard.postbuf = *postbuf; +                        } + +                        local->success_count++; + +                        if ((local->success_count >= priv->wait_count) +                            && local->read_child_returned) { +                                need_unwind = 1; +                        } +                } +                local->op_errno = op_errno; +        } +        UNLOCK (&frame->lock); + +        if (need_unwind) +                local->transaction.unwind (frame, this); + +        call_count = afr_frame_return (frame); + +        if (call_count == 0) { +                local->transaction.resume (frame, this); +        } + +        return 0; +} + +static int +afr_discard_wind (call_frame_t *frame, xlator_t *this) +{ +        afr_local_t *local = NULL; +        afr_private_t *priv = NULL; +        int call_count = -1; +        int i = 0; + +        local = frame->local; +        priv = this->private; + +        call_count = afr_pre_op_done_children_count (local->transaction.pre_op, +                                                     priv->child_count); + +        if (call_count == 0) { +                local->transaction.resume (frame, this); +                return 0; +        } + +        local->call_count = call_count; + +        for (i = 0; i < priv->child_count; i++) { +                if (local->transaction.pre_op[i]) { +                        STACK_WIND_COOKIE (frame, afr_discard_wind_cbk, +                                           (void *) (long) i, +                                           priv->children[i], +                                           priv->children[i]->fops->discard, +                                           local->fd, +                                           local->cont.discard.offset, +					   local->cont.discard.len, +                                           NULL); + +                        if (!--call_count) +                                break; +                } +        } + +        return 0; +} + +static int +afr_discard_done (call_frame_t *frame, xlator_t *this) +{ +        afr_local_t *local = NULL; + +        local = frame->local; + +        local->transaction.unwind (frame, this); + +        AFR_STACK_DESTROY (frame); + +        return 0; +} + +static int +afr_do_discard(call_frame_t *frame, xlator_t *this) +{ +        call_frame_t * transaction_frame = NULL; +        afr_local_t *  local             = NULL; +        int op_ret   = -1; +        int op_errno = 0; + +        local = frame->local; + +        transaction_frame = copy_frame (frame); +        if (!transaction_frame) { +                goto out; +        } + +        transaction_frame->local = local; +        frame->local = NULL; + +        local->op = GF_FOP_DISCARD; + +        local->transaction.fop    = afr_discard_wind; +        local->transaction.done   = afr_discard_done; +        local->transaction.unwind = afr_discard_unwind; + +        local->transaction.main_frame = frame; + +        local->transaction.start   = local->cont.discard.offset; +        local->transaction.len     = 0; + +        op_ret = afr_transaction (transaction_frame, this, AFR_DATA_TRANSACTION); +        if (op_ret < 0) { +            op_errno = -op_ret; +            goto out; +        } + +        op_ret = 0; +out: +        if (op_ret < 0) { +                if (transaction_frame) +                        AFR_STACK_DESTROY (transaction_frame); +                AFR_STACK_UNWIND (discard, frame, op_ret, op_errno, NULL, +                                  NULL, NULL); +        } + +        return 0; +} + +int +afr_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	    size_t len, dict_t *xdata) +{ +        afr_private_t * priv  = NULL; +        afr_local_t   * local = NULL; +        call_frame_t   *transaction_frame = NULL; +        int ret = -1; +        int op_errno = 0; + +        VALIDATE_OR_GOTO (frame, out); +        VALIDATE_OR_GOTO (this, out); +        VALIDATE_OR_GOTO (this->private, out); + +        priv = this->private; + +        if (afr_is_split_brain (this, fd->inode)) { +                op_errno = EIO; +                goto out; +        } +        QUORUM_CHECK(discard, out); + +        AFR_LOCAL_ALLOC_OR_GOTO (frame->local, out); +        local = frame->local; + +        ret = afr_local_init (local, priv, &op_errno); +        if (ret < 0) +                goto out; + +        local->cont.discard.offset  = offset; +	local->cont.discard.len = len; + +        local->fd = fd_ref (fd); + +        afr_open_fd_fix (fd, this); + +        afr_do_discard(frame, this); + +        ret = 0; +out: +        if (ret < 0) { +                if (transaction_frame) +                        AFR_STACK_DESTROY (transaction_frame); +                AFR_STACK_UNWIND (discard, frame, -1, op_errno, NULL, NULL, NULL); +        } + +        return 0; +} + +/* }}} */ +  /* {{{ fsync */  int32_t diff --git a/xlators/cluster/afr/src/afr.c b/xlators/cluster/afr/src/afr.c index 4d6c714801f..6fabc00171f 100644 --- a/xlators/cluster/afr/src/afr.c +++ b/xlators/cluster/afr/src/afr.c @@ -484,6 +484,7 @@ struct xlator_fops fops = {          .entrylk     = afr_entrylk,          .fentrylk    = afr_fentrylk,  	.fallocate   = afr_fallocate, +	.discard     = afr_discard,          /* inode read */          .access      = afr_access, diff --git a/xlators/cluster/afr/src/afr.h b/xlators/cluster/afr/src/afr.h index 185f4f7134a..ced4e6fab25 100644 --- a/xlators/cluster/afr/src/afr.h +++ b/xlators/cluster/afr/src/afr.h @@ -660,6 +660,14 @@ typedef struct _afr_local {  			struct iatt prebuf;  			struct iatt postbuf;  		} fallocate; + +		struct { +			off_t offset; +			size_t len; +			struct iatt prebuf; +			struct iatt postbuf; +		} discard; +          } cont;          struct { diff --git a/xlators/cluster/dht/src/dht-common.h b/xlators/cluster/dht/src/dht-common.h index de35e843f47..fb90e48ccb0 100644 --- a/xlators/cluster/dht/src/dht-common.h +++ b/xlators/cluster/dht/src/dht-common.h @@ -693,6 +693,8 @@ int32_t dht_fsetattr (call_frame_t *frame, xlator_t *this, fd_t *fd,                        struct iatt  *stbuf, int32_t valid, dict_t *xdata);  int32_t dht_fallocate(call_frame_t *frame, xlator_t *this, fd_t *fd,  		      int32_t mode, off_t offset, size_t len, dict_t *xdata); +int32_t dht_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, +		    off_t offset, size_t len, dict_t *xdata);  int32_t dht_init (xlator_t *this);  void    dht_fini (xlator_t *this); diff --git a/xlators/cluster/dht/src/dht-inode-write.c b/xlators/cluster/dht/src/dht-inode-write.c index 56ed36ef0a9..9bcd84ae15a 100644 --- a/xlators/cluster/dht/src/dht-inode-write.c +++ b/xlators/cluster/dht/src/dht-inode-write.c @@ -20,6 +20,7 @@ int dht_writev2 (xlator_t *this, call_frame_t *frame, int ret);  int dht_truncate2 (xlator_t *this, call_frame_t *frame, int ret);  int dht_setattr2 (xlator_t *this, call_frame_t *frame, int ret);  int dht_fallocate2(xlator_t *this, call_frame_t *frame, int op_ret); +int dht_discard2(xlator_t *this, call_frame_t *frame, int op_ret);  int  dht_writev_cbk (call_frame_t *frame, void *cookie, xlator_t *this, @@ -488,6 +489,141 @@ err:  } +int +dht_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +                int op_ret, int op_errno, struct iatt *prebuf, +                struct iatt *postbuf, dict_t *xdata) +{ +        dht_local_t  *local = NULL; +        call_frame_t *prev = NULL; +        int           ret = -1; + +        GF_VALIDATE_OR_GOTO ("dht", frame, err); +        GF_VALIDATE_OR_GOTO ("dht", this, out); +        GF_VALIDATE_OR_GOTO ("dht", frame->local, out); +        GF_VALIDATE_OR_GOTO ("dht", cookie, out); + +        local = frame->local; +        prev = cookie; + +        if ((op_ret == -1) && (op_errno != ENOENT)) { +                local->op_errno = op_errno; +                local->op_ret = -1; +                gf_log (this->name, GF_LOG_DEBUG, +                        "subvolume %s returned -1 (%s)", +                        prev->this->name, strerror (op_errno)); + +                goto out; +        } + +        if (local->call_cnt != 1) { +                if (local->stbuf.ia_blocks) { +                        dht_iatt_merge (this, postbuf, &local->stbuf, NULL); +                        dht_iatt_merge (this, prebuf, &local->prebuf, NULL); +                } +                goto out; +        } +        local->rebalance.target_op_fn = dht_discard2; + +        /* Phase 2 of migration */ +        if ((op_ret == -1) || IS_DHT_MIGRATION_PHASE2 (postbuf)) { +                ret = dht_rebalance_complete_check (this, frame); +                if (!ret) +                        return 0; +        } + +        /* Check if the rebalance phase1 is true */ +        if (IS_DHT_MIGRATION_PHASE1 (postbuf)) { +                dht_iatt_merge (this, &local->stbuf, postbuf, NULL); +                dht_iatt_merge (this, &local->prebuf, prebuf, NULL); +                ret = fd_ctx_get (local->fd, this, NULL); +                if (!ret) { +                        dht_discard2 (this, frame, 0); +                        return 0; +                } +                ret = dht_rebalance_in_progress_check (this, frame); +                if (!ret) +                        return 0; +        } + +out: +        DHT_STRIP_PHASE1_FLAGS (postbuf); +        DHT_STRIP_PHASE1_FLAGS (prebuf); +        DHT_STACK_UNWIND (discard, frame, op_ret, op_errno, +                          prebuf, postbuf, xdata); +err: +        return 0; +} + +int +dht_discard2(xlator_t *this, call_frame_t *frame, int op_ret) +{ +        dht_local_t  *local  = NULL; +        xlator_t     *subvol = NULL; +        uint64_t      tmp_subvol = 0; +        int           ret = -1; + +        local = frame->local; + +        if (local->fd) +                ret = fd_ctx_get (local->fd, this, &tmp_subvol); +        if (!ret) +                subvol = (xlator_t *)(long)tmp_subvol; + +        if (!subvol) +                subvol = local->cached_subvol; + +        local->call_cnt = 2; /* This is the second attempt */ + +	STACK_WIND(frame, dht_discard_cbk, subvol, subvol->fops->discard, +		   local->fd, local->rebalance.offset, local->rebalance.size, +		   NULL); + +        return 0; +} + +int +dht_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	    size_t len, dict_t *xdata) +{ +        xlator_t     *subvol = NULL; +        int           op_errno = -1; +        dht_local_t  *local = NULL; + +        VALIDATE_OR_GOTO (frame, err); +        VALIDATE_OR_GOTO (this, err); +        VALIDATE_OR_GOTO (fd, err); + +        local = dht_local_init (frame, NULL, fd, GF_FOP_DISCARD); +        if (!local) { +                op_errno = ENOMEM; +                goto err; +        } + +        local->rebalance.offset = offset; +	local->rebalance.size = len; + +        local->call_cnt = 1; +        subvol = local->cached_subvol; +        if (!subvol) { +                gf_log (this->name, GF_LOG_DEBUG, +                        "no cached subvolume for fd=%p", fd); +                op_errno = EINVAL; +                goto err; +        } + +        STACK_WIND (frame, dht_discard_cbk, subvol, subvol->fops->discard, +                    fd, offset, len, xdata); + +        return 0; + +err: +        op_errno = (op_errno == -1) ? errno : op_errno; +        DHT_STACK_UNWIND (discard, frame, -1, op_errno, NULL, NULL, NULL); + +        return 0; +} +  /* handle cases of migration here for 'setattr()' calls */  int  dht_file_setattr_cbk (call_frame_t *frame, void *cookie, xlator_t *this, diff --git a/xlators/cluster/dht/src/dht.c b/xlators/cluster/dht/src/dht.c index d3031e2032b..0349b63a91f 100644 --- a/xlators/cluster/dht/src/dht.c +++ b/xlators/cluster/dht/src/dht.c @@ -71,6 +71,7 @@ struct xlator_fops fops = {          .setattr     = dht_setattr,          .fsetattr    = dht_fsetattr,  	.fallocate   = dht_fallocate, +	.discard     = dht_discard,  };  struct xlator_dumpops dumpops = { diff --git a/xlators/cluster/stripe/src/stripe.c b/xlators/cluster/stripe/src/stripe.c index 3c56464b8c1..c1294bdd421 100644 --- a/xlators/cluster/stripe/src/stripe.c +++ b/xlators/cluster/stripe/src/stripe.c @@ -3941,6 +3941,172 @@ err:  int32_t +stripe_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +                   int32_t op_ret, int32_t op_errno, struct iatt *prebuf, +                   struct iatt *postbuf, dict_t *xdata) +{ +        int32_t         callcnt = 0; +        stripe_local_t *local = NULL; +	stripe_local_t *mlocal = NULL; +        call_frame_t   *prev = NULL; +	call_frame_t   *mframe = NULL; + +        if (!this || !frame || !frame->local || !cookie) { +                gf_log ("stripe", GF_LOG_DEBUG, "possible NULL deref"); +                goto out; +        } + +        prev  = cookie; +        local = frame->local; +	mframe = local->orig_frame; +	mlocal = mframe->local; + +        LOCK(&frame->lock); +        { +                callcnt = ++mlocal->call_count; + +                if (op_ret == 0) { +                        mlocal->post_buf = *postbuf; +                        mlocal->pre_buf = *prebuf; + +			mlocal->prebuf_blocks  += prebuf->ia_blocks; +			mlocal->postbuf_blocks += postbuf->ia_blocks; + +			correct_file_size(prebuf, mlocal->fctx, prev); +			correct_file_size(postbuf, mlocal->fctx, prev); + +			if (mlocal->prebuf_size < prebuf->ia_size) +				mlocal->prebuf_size = prebuf->ia_size; +			if (mlocal->postbuf_size < postbuf->ia_size) +				mlocal->postbuf_size = postbuf->ia_size; +                } + +		/* return the first failure */ +		if (mlocal->op_ret == 0) { +			mlocal->op_ret = op_ret; +			mlocal->op_errno = op_errno; +		} +        } +        UNLOCK (&frame->lock); + +        if ((callcnt == mlocal->wind_count) && mlocal->unwind) { +		mlocal->pre_buf.ia_size = mlocal->prebuf_size; +		mlocal->pre_buf.ia_blocks = mlocal->prebuf_blocks; +		mlocal->post_buf.ia_size = mlocal->postbuf_size; +		mlocal->post_buf.ia_blocks = mlocal->postbuf_blocks; + +                STRIPE_STACK_UNWIND (discard, mframe, mlocal->op_ret, +                                     mlocal->op_errno, &mlocal->pre_buf, +                                     &mlocal->post_buf, NULL); +        } +out: +	STRIPE_STACK_DESTROY(frame); +        return 0; +} + +int32_t +stripe_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	       size_t len, dict_t *xdata) +{ +        stripe_local_t   *local = NULL; +        stripe_fd_ctx_t  *fctx = NULL; +        int32_t           op_errno = 1; +        int32_t           idx = 0; +        int32_t           offset_offset = 0; +        int32_t           remaining_size = 0; +        off_t             fill_size = 0; +        uint64_t          stripe_size = 0; +        uint64_t          tmp_fctx = 0; +	off_t		  dest_offset = 0; +	call_frame_t	  *fframe = NULL; +	stripe_local_t	  *flocal = NULL; + +        VALIDATE_OR_GOTO (frame, err); +        VALIDATE_OR_GOTO (this, err); +        VALIDATE_OR_GOTO (fd, err); +        VALIDATE_OR_GOTO (fd->inode, err); + +        inode_ctx_get (fd->inode, this, &tmp_fctx); +        if (!tmp_fctx) { +                op_errno = EINVAL; +                goto err; +        } +        fctx = (stripe_fd_ctx_t *)(long)tmp_fctx; +        stripe_size = fctx->stripe_size; + +        STRIPE_VALIDATE_FCTX (fctx, err); + +        remaining_size = len; + +        local = mem_get0 (this->local_pool); +        if (!local) { +                op_errno = ENOMEM; +                goto err; +        } +        frame->local = local; +        local->stripe_size = stripe_size; +	local->fctx = fctx; + +        if (!stripe_size) { +                gf_log (this->name, GF_LOG_DEBUG, +                        "Wrong stripe size for the file"); +                op_errno = EINVAL; +                goto err; +        } + +        while (1) { +		fframe = copy_frame(frame); +		flocal = mem_get0(this->local_pool); +		if (!flocal) { +			op_errno = ENOMEM; +			goto err; +		} +		flocal->orig_frame = frame; +		fframe->local = flocal; + +		/* send discard request to the associated child node */ +                idx = (((offset + offset_offset) / +                        local->stripe_size) % fctx->stripe_count); + +                fill_size = (local->stripe_size - +                             ((offset + offset_offset) % local->stripe_size)); +                if (fill_size > remaining_size) +                        fill_size = remaining_size; + +                remaining_size -= fill_size; + +                local->wind_count++; +                if (remaining_size == 0) +                        local->unwind = 1; + +		dest_offset = offset + offset_offset; +		if (fctx->stripe_coalesce) +			dest_offset = coalesced_offset(dest_offset, +					local->stripe_size, fctx->stripe_count); + +		/* +		 * TODO: Create a separate handler for coalesce mode that sends a +		 * single discard per-child (since the ranges are linear). +		 */ +		STACK_WIND(fframe, stripe_discard_cbk, fctx->xl_array[idx], +			   fctx->xl_array[idx]->fops->discard, fd, dest_offset, +			   fill_size, xdata); + +                offset_offset += fill_size; +                if (remaining_size == 0) +                        break; +        } + +        return 0; +err: +	if (fframe) +		STRIPE_STACK_DESTROY(fframe); + +        STRIPE_STACK_UNWIND (discard, frame, -1, op_errno, NULL, NULL, NULL); +        return 0; +} + +int32_t  stripe_release (xlator_t *this, fd_t *fd)  {  	return 0; @@ -5389,6 +5555,7 @@ struct xlator_fops fops = {          .fremovexattr   = stripe_fremovexattr,          .readdirp       = stripe_readdirp,  	.fallocate	= stripe_fallocate, +	.discard	= stripe_discard,  };  struct xlator_cbks cbks = { diff --git a/xlators/debug/io-stats/src/io-stats.c b/xlators/debug/io-stats/src/io-stats.c index 5bb0e9d618c..4f27a2e418f 100644 --- a/xlators/debug/io-stats/src/io-stats.c +++ b/xlators/debug/io-stats/src/io-stats.c @@ -1736,6 +1736,18 @@ io_stats_fallocate_cbk(call_frame_t *frame, void *cookie, xlator_t *this,  int +io_stats_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +		     int32_t op_ret, int32_t op_errno, struct iatt *prebuf, +		     struct iatt *postbuf, dict_t *xdata) +{ +	UPDATE_PROFILE_STATS(frame, DISCARD); +	STACK_UNWIND_STRICT(discard, frame, op_ret, op_errno, prebuf, postbuf, +			    xdata); +	return 0; +} + + +int  io_stats_lk_cbk (call_frame_t *frame, void *cookie, xlator_t *this,                   int32_t op_ret, int32_t op_errno, struct gf_flock *lock, dict_t *xdata)  { @@ -2416,6 +2428,20 @@ io_stats_fallocate(call_frame_t *frame, xlator_t *this, fd_t *fd, int32_t mode,  	return 0;  } + +int +io_stats_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +		 size_t len, dict_t *xdata) +{ +	START_FOP_LATENCY(frame); + +	STACK_WIND(frame, io_stats_discard_cbk, FIRST_CHILD(this), +		   FIRST_CHILD(this)->fops->discard, fd, offset, len, xdata); + +	return 0; +} + +  int  io_stats_lk (call_frame_t *frame, xlator_t *this,               fd_t *fd, int32_t cmd, struct gf_flock *lock, dict_t *xdata) @@ -2843,6 +2869,7 @@ struct xlator_fops fops = {          .setattr     = io_stats_setattr,          .fsetattr    = io_stats_fsetattr,  	.fallocate   = io_stats_fallocate, +	.discard     = io_stats_discard,  };  struct xlator_cbks cbks = { diff --git a/xlators/features/marker/src/marker.c b/xlators/features/marker/src/marker.c index 2dc03f93195..112e71dff31 100644 --- a/xlators/features/marker/src/marker.c +++ b/xlators/features/marker/src/marker.c @@ -1876,6 +1876,75 @@ err:          return 0;  } + +int32_t +marker_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +                   int32_t op_ret, int32_t op_errno, struct iatt *prebuf, +                   struct iatt *postbuf, dict_t *xdata) +{ +        marker_local_t     *local   = NULL; +        marker_conf_t      *priv    = NULL; + +        if (op_ret == -1) { +                gf_log (this->name, GF_LOG_TRACE, "%s occurred during discard", +                        strerror (op_errno)); +        } + +        local = (marker_local_t *) frame->local; + +        frame->local = NULL; + +        STACK_UNWIND_STRICT (discard, frame, op_ret, op_errno, prebuf, +                             postbuf, xdata); + +        if (op_ret == -1 || local == NULL) +                goto out; + +        priv = this->private; + +        if (priv->feature_enabled & GF_QUOTA) +                mq_initiate_quota_txn (this, &local->loc); + +        if (priv->feature_enabled & GF_XTIME) +                marker_xtime_update_marks (this, local); +out: +        marker_local_unref (local); + +        return 0; +} + +int32_t +marker_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	       size_t len, dict_t *xdata) +{ +        int32_t          ret   = 0; +        marker_local_t  *local = NULL; +        marker_conf_t   *priv  = NULL; + +        priv = this->private; + +        if (priv->feature_enabled == 0) +                goto wind; + +        local = mem_get0 (this->local_pool); + +        MARKER_INIT_LOCAL (frame, local); + +        ret = marker_inode_loc_fill (fd->inode, &local->loc); + +        if (ret == -1) +                goto err; +wind: +        STACK_WIND (frame, marker_discard_cbk, FIRST_CHILD(this), +                    FIRST_CHILD(this)->fops->discard, fd, offset, len, xdata); +        return 0; +err: +        STACK_UNWIND_STRICT (discard, frame, -1, ENOMEM, NULL, NULL, NULL); + +        return 0; +} + +  /* when a call from the special client is received on   * key trusted.glusterfs.volume-mark with value "RESET"   * or if the value is 0length, update the change the @@ -2686,6 +2755,7 @@ struct xlator_fops fops = {          .getxattr    = marker_getxattr,          .readdirp    = marker_readdirp,  	.fallocate   = marker_fallocate, +	.discard     = marker_discard,  };  struct xlator_cbks cbks = { diff --git a/xlators/mount/fuse/src/fuse-bridge.c b/xlators/mount/fuse/src/fuse-bridge.c index 59472b91485..1b97d4302c4 100644 --- a/xlators/mount/fuse/src/fuse-bridge.c +++ b/xlators/mount/fuse/src/fuse-bridge.c @@ -2669,10 +2669,13 @@ fuse_fallocate_resume(fuse_state_t *state)  	       state->finh->unique, state->fd, state->flags, state->size,  	       state->off); -	/* we only support KEEP_SIZE */ -	FUSE_FOP(state, fuse_fallocate_cbk, GF_FOP_FALLOCATE, fallocate, state->fd, -		 (state->flags & FALLOC_FL_KEEP_SIZE), state->off, state->size, -		 state->xdata); +	if (state->flags & FALLOC_FL_PUNCH_HOLE) +		FUSE_FOP(state, fuse_fallocate_cbk, GF_FOP_DISCARD, discard, +			 state->fd, state->off, state->size, state->xdata); +	else +		FUSE_FOP(state, fuse_fallocate_cbk, GF_FOP_FALLOCATE, fallocate, +			 state->fd, (state->flags & FALLOC_FL_KEEP_SIZE), +			 state->off, state->size, state->xdata);  }  static void @@ -2687,13 +2690,6 @@ fuse_fallocate(xlator_t *this, fuse_in_header_t *finh, void *msg)  	state->flags = ffi->mode;  	state->fd = FH_TO_FD(ffi->fh); -	/* discard TBD as separate FOP */ -	if (state->flags & FALLOC_FL_PUNCH_HOLE) { -		send_fuse_err(this, finh, EOPNOTSUPP); -		free_fuse_state(state); -		return; -	} -  	fuse_resolve_fd_init(state, &state->resolve, state->fd);  	fuse_resolve_and_resume(state, fuse_fallocate_resume);  } diff --git a/xlators/performance/io-cache/src/io-cache.c b/xlators/performance/io-cache/src/io-cache.c index 30dc14a9cdb..d0667e46944 100644 --- a/xlators/performance/io-cache/src/io-cache.c +++ b/xlators/performance/io-cache/src/io-cache.c @@ -1422,6 +1422,33 @@ ioc_readdirp (call_frame_t *frame, xlator_t *this, fd_t *fd, size_t size,          return 0;  } +static int32_t +ioc_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +		int32_t op_ret, int32_t op_errno, struct iatt *pre, +		struct iatt *post, dict_t *xdata) +{ +	STACK_UNWIND_STRICT(discard, frame, op_ret, op_errno, pre, post, xdata); +	return 0; +} + +static int32_t +ioc_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	    size_t len, dict_t *xdata) +{ +	uint64_t ioc_inode = 0; + +	inode_ctx_get (fd->inode, this, &ioc_inode); + +	if (ioc_inode) +		ioc_inode_flush ((ioc_inode_t *)(long)ioc_inode); + +	STACK_WIND(frame, ioc_discard_cbk, FIRST_CHILD(this), +		   FIRST_CHILD(this)->fops->discard, fd, offset, len, xdata); +       return 0; +} + + +  int32_t  ioc_get_priority_list (const char *opt_str, struct list_head *first)  { @@ -2044,6 +2071,7 @@ struct xlator_fops fops = {          .mknod       = ioc_mknod,          .readdirp    = ioc_readdirp, +	.discard     = ioc_discard,  }; diff --git a/xlators/performance/io-threads/src/io-threads.c b/xlators/performance/io-threads/src/io-threads.c index 33642bffb25..c6bdc375439 100644 --- a/xlators/performance/io-threads/src/io-threads.c +++ b/xlators/performance/io-threads/src/io-threads.c @@ -308,6 +308,7 @@ iot_schedule (call_frame_t *frame, xlator_t *this, call_stub_t *stub)          case GF_FOP_FXATTROP:          case GF_FOP_RCHECKSUM:  	case GF_FOP_FALLOCATE: +	case GF_FOP_DISCARD:                  pri = IOT_PRI_LO;                  break; @@ -2459,6 +2460,56 @@ out:  }  int +iot_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +                  int32_t op_ret, int32_t op_errno, +                  struct iatt *preop, struct iatt *postop, dict_t *xdata) +{ +        STACK_UNWIND_STRICT (discard, frame, op_ret, op_errno, preop, postop, +                             xdata); +        return 0; +} + + +int +iot_discard_wrapper(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +		    size_t len, dict_t *xdata) +{ +        STACK_WIND (frame, iot_discard_cbk, FIRST_CHILD (this), +                    FIRST_CHILD (this)->fops->discard, fd, offset, len, xdata); +        return 0; +} + + +int +iot_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	    size_t len, dict_t *xdata) +{ +        call_stub_t     *stub = NULL; +        int              ret = -1; + +        stub = fop_discard_stub(frame, iot_discard_wrapper, fd, offset, len, +				xdata); +        if (!stub) { +                gf_log (this->name, GF_LOG_ERROR, "cannot create discard stub" +                        "(out of memory)"); +                ret = -ENOMEM; +                goto out; +        } + +        ret = iot_schedule (frame, this, stub); + +out: +        if (ret < 0) { +                STACK_UNWIND_STRICT (discard, frame, -1, -ret, NULL, NULL, +                                     NULL); +                if (stub != NULL) { +                        call_stub_destroy (stub); +                } +        } +        return 0; +} + +int  __iot_workers_scale (iot_conf_t *conf)  {          int       scale = 0; @@ -2788,6 +2839,7 @@ struct xlator_fops fops = {  	.fxattrop    = iot_fxattrop,          .rchecksum   = iot_rchecksum,  	.fallocate   = iot_fallocate, +	.discard     = iot_discard,  };  struct xlator_cbks cbks; diff --git a/xlators/performance/md-cache/src/md-cache.c b/xlators/performance/md-cache/src/md-cache.c index 0b2c94319fb..2f52cbe5807 100644 --- a/xlators/performance/md-cache/src/md-cache.c +++ b/xlators/performance/md-cache/src/md-cache.c @@ -1889,6 +1889,45 @@ int mdc_fallocate(call_frame_t *frame, xlator_t *this, fd_t *fd, int32_t mode,  }  int +mdc_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +                int32_t op_ret, int32_t op_errno, +                struct iatt *prebuf, struct iatt *postbuf, dict_t *xdata) +{ +        mdc_local_t  *local = NULL; + +        local = frame->local; + +        if (op_ret != 0) +                goto out; + +        if (!local) +                goto out; + +        mdc_inode_iatt_set_validate(this, local->fd->inode, prebuf, postbuf); + +out: +        MDC_STACK_UNWIND(discard, frame, op_ret, op_errno, prebuf, postbuf, +                         xdata); + +        return 0; +} + +int mdc_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +		size_t len, dict_t *xdata) +{ +	mdc_local_t *local; + +	local = mdc_local_get(frame); +	local->fd = fd_ref(fd); + +	STACK_WIND(frame, mdc_discard_cbk, FIRST_CHILD(this), +		   FIRST_CHILD(this)->fops->discard, fd, offset, len, +		   xdata); + +	return 0; +} + +int  mdc_forget (xlator_t *this, inode_t *inode)  {          mdc_inode_wipe (this, inode); @@ -2017,6 +2056,7 @@ struct xlator_fops fops = {  	.readdirp    = mdc_readdirp,  	.readdir     = mdc_readdir,  	.fallocate   = mdc_fallocate, +	.discard     = mdc_discard,  }; diff --git a/xlators/performance/open-behind/src/open-behind.c b/xlators/performance/open-behind/src/open-behind.c index 5ac813132eb..26b684e888c 100644 --- a/xlators/performance/open-behind/src/open-behind.c +++ b/xlators/performance/open-behind/src/open-behind.c @@ -701,6 +701,25 @@ err:  }  int +ob_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	   size_t len, dict_t *xdata) +{ +	call_stub_t *stub; + +	stub = fop_discard_stub(frame, default_discard_resume, fd, offset, len, +				xdata); +	if (!stub) +		goto err; + +	open_and_resume(this, fd, stub); + +	return 0; +err: +	STACK_UNWIND_STRICT(discard, frame, -1, ENOMEM, NULL, NULL, NULL); +	return 0; +} + +int  ob_unlink (call_frame_t *frame, xlator_t *this, loc_t *loc, int xflags,  	   dict_t *xdata)  { @@ -922,6 +941,7 @@ struct xlator_fops fops = {  	.fxattrop    = ob_fxattrop,  	.fsetattr    = ob_fsetattr,  	.fallocate   = ob_fallocate, +	.discard     = ob_discard,  	.unlink      = ob_unlink,  	.rename      = ob_rename,  	.lk          = ob_lk, diff --git a/xlators/performance/read-ahead/src/read-ahead.c b/xlators/performance/read-ahead/src/read-ahead.c index 522fa52b8e9..241fa477fda 100644 --- a/xlators/performance/read-ahead/src/read-ahead.c +++ b/xlators/performance/read-ahead/src/read-ahead.c @@ -942,6 +942,55 @@ unwind:          return 0;  } +int +ra_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +               int32_t op_ret, int32_t op_errno, struct iatt *prebuf, +               struct iatt *postbuf, dict_t *xdata) +{ +        GF_ASSERT (frame); + +        STACK_UNWIND_STRICT (discard, frame, op_ret, op_errno, prebuf, +                             postbuf, xdata); +        return 0; +} + +static int +ra_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	     size_t len, dict_t *xdata) +{ +        ra_file_t *file    = NULL; +        fd_t      *iter_fd = NULL; +        inode_t   *inode   = NULL; +        uint64_t  tmp_file = 0; +        int32_t   op_errno = EINVAL; + +        GF_ASSERT (frame); +        GF_VALIDATE_OR_GOTO (frame->this->name, this, unwind); +        GF_VALIDATE_OR_GOTO (frame->this->name, fd, unwind); + +        inode = fd->inode; + +        LOCK (&inode->lock); +        { +                list_for_each_entry (iter_fd, &inode->fd_list, inode_list) { +                        fd_ctx_get (iter_fd, this, &tmp_file); +                        file = (ra_file_t *)(long)tmp_file; +                        if (!file) +                                continue; + +                        flush_region(frame, file, offset, len, 1); +                } +        } +        UNLOCK (&inode->lock); + +        STACK_WIND (frame, ra_discard_cbk, FIRST_CHILD (this), +                    FIRST_CHILD (this)->fops->discard, fd, offset, len, xdata); +        return 0; + +unwind: +        STACK_UNWIND_STRICT (discard, frame, -1, op_errno, NULL, NULL, NULL); +        return 0; +}  int  ra_priv_dump (xlator_t *this) @@ -1123,6 +1172,7 @@ struct xlator_fops fops = {          .truncate    = ra_truncate,          .ftruncate   = ra_ftruncate,          .fstat       = ra_fstat, +	.discard     = ra_discard,  };  struct xlator_cbks cbks = { diff --git a/xlators/protocol/client/src/client-rpc-fops.c b/xlators/protocol/client/src/client-rpc-fops.c index 53cf3968a2b..c055428340c 100644 --- a/xlators/protocol/client/src/client-rpc-fops.c +++ b/xlators/protocol/client/src/client-rpc-fops.c @@ -1993,6 +1993,61 @@ out:  }  int +client3_3_discard_cbk(struct rpc_req *req, struct iovec *iov, int count, +		      void *myframe) +{ +        call_frame_t    *frame      = NULL; +        gfs3_discard_rsp rsp      = {0,}; +        struct iatt      prestat    = {0,}; +        struct iatt      poststat   = {0,}; +        int              ret        = 0; +        xlator_t *this       = NULL; +        dict_t  *xdata       = NULL; + +        this = THIS; + +        frame = myframe; + +        if (-1 == req->rpc_status) { +                rsp.op_ret   = -1; +                rsp.op_errno = ENOTCONN; +                goto out; +        } +        ret = xdr_to_generic(*iov, &rsp, (xdrproc_t) xdr_gfs3_discard_rsp); +        if (ret < 0) { +                gf_log (this->name, GF_LOG_ERROR, "XDR decoding failed"); +                rsp.op_ret   = -1; +                rsp.op_errno = EINVAL; +                goto out; +        } + +        if (-1 != rsp.op_ret) { +                gf_stat_to_iatt (&rsp.statpre, &prestat); +                gf_stat_to_iatt (&rsp.statpost, &poststat); +        } + +        GF_PROTOCOL_DICT_UNSERIALIZE (this, xdata, (rsp.xdata.xdata_val), +                                      (rsp.xdata.xdata_len), ret, +                                      rsp.op_errno, out); + +out: +        if (rsp.op_ret == -1) { +                gf_log (this->name, GF_LOG_WARNING, "remote operation failed: %s", +                        strerror (gf_error_to_errno (rsp.op_errno))); +        } +        CLIENT_STACK_UNWIND (discard, frame, rsp.op_ret, +                             gf_error_to_errno (rsp.op_errno), &prestat, +                             &poststat, xdata); + +        free (rsp.xdata.xdata_val); + +        if (xdata) +                dict_unref (xdata); + +        return 0; +} + +int  client3_3_setattr_cbk (struct rpc_req *req, struct iovec *iov, int count,                         void *myframe)  { @@ -5888,6 +5943,50 @@ unwind:          return 0;  } +int32_t +client3_3_discard(call_frame_t *frame, xlator_t *this, void *data) +{ +        clnt_args_t       *args     = NULL; +        int64_t            remote_fd = -1; +        clnt_conf_t       *conf     = NULL; +	gfs3_discard_req   req	    = {{0},}; +        int                op_errno = ESTALE; +        int                ret        = 0; + +        if (!frame || !this || !data) +                goto unwind; + +        args = data; +        conf = this->private; + +        CLIENT_GET_REMOTE_FD (this, args->fd, DEFAULT_REMOTE_FD, +                              remote_fd, op_errno, unwind); + +        req.fd = remote_fd; +	req.offset = args->offset; +	req.size = args->size; +	memcpy(req.gfid, args->fd->inode->gfid, 16); + +        GF_PROTOCOL_DICT_SERIALIZE (this, args->xdata, (&req.xdata.xdata_val), +                                    req.xdata.xdata_len, op_errno, unwind); + +        ret = client_submit_request(this, &req, frame, conf->fops, +                                    GFS3_OP_DISCARD, client3_3_discard_cbk, +				    NULL, NULL, 0, NULL, 0, NULL, +                                    (xdrproc_t) xdr_gfs3_discard_req); +        if (ret) +                gf_log (this->name, GF_LOG_WARNING, "failed to send the fop"); + +        GF_FREE (req.xdata.xdata_val); + +        return 0; +unwind: +        CLIENT_STACK_UNWIND(discard, frame, -1, op_errno, NULL, NULL, NULL); +        GF_FREE (req.xdata.xdata_val); + +        return 0; +} +  /* Table Specific to FOPS */ @@ -5934,6 +6033,7 @@ rpc_clnt_procedure_t clnt3_3_fop_actors[GF_FOP_MAXVALUE] = {          [GF_FOP_FSETATTR]    = { "FSETATTR",    client3_3_fsetattr },          [GF_FOP_READDIRP]    = { "READDIRP",    client3_3_readdirp },  	[GF_FOP_FALLOCATE]   = { "FALLOCATE",	client3_3_fallocate }, +	[GF_FOP_DISCARD]     = { "DISCARD",	client3_3_discard },          [GF_FOP_RELEASE]     = { "RELEASE",     client3_3_release },          [GF_FOP_RELEASEDIR]  = { "RELEASEDIR",  client3_3_releasedir },          [GF_FOP_GETSPEC]     = { "GETSPEC",     client3_getspec }, @@ -5984,6 +6084,7 @@ char *clnt3_3_fop_names[GFS3_OP_MAXVALUE] = {          [GFS3_OP_FSETATTR]    = "FSETATTR",          [GFS3_OP_READDIRP]    = "READDIRP",  	[GFS3_OP_FALLOCATE]   = "FALLOCATE", +	[GFS3_OP_DISCARD]     = "DISCARD",          [GFS3_OP_RELEASE]     = "RELEASE",          [GFS3_OP_RELEASEDIR]  = "RELEASEDIR",          [GFS3_OP_FREMOVEXATTR] = "FREMOVEXATTR", diff --git a/xlators/protocol/client/src/client.c b/xlators/protocol/client/src/client.c index 377263af64a..aa2b8eef8e5 100644 --- a/xlators/protocol/client/src/client.c +++ b/xlators/protocol/client/src/client.c @@ -1997,6 +1997,40 @@ out:  }  int32_t +client_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	       size_t len, dict_t *xdata) +{ +        int          ret  = -1; +        clnt_conf_t *conf = NULL; +        rpc_clnt_procedure_t *proc = NULL; +        clnt_args_t  args = {0,}; + +        conf = this->private; +        if (!conf || !conf->fops) +                goto out; + +        args.fd = fd; +	args.offset = offset; +	args.size = len; +        args.xdata = xdata; + +        proc = &conf->fops->proctable[GF_FOP_DISCARD]; +        if (!proc) { +                gf_log (this->name, GF_LOG_ERROR, +                        "rpc procedure not found for %s", +                        gf_fop_list[GF_FOP_DISCARD]); +                goto out; +        } +        if (proc->fn) +                ret = proc->fn (frame, this, &args); +out: +        if (ret) +                STACK_UNWIND_STRICT(discard, frame, -1, ENOTCONN, NULL, NULL, NULL); + +	return 0; +} + +int32_t  client_getspec (call_frame_t *frame, xlator_t *this, const char *key,                  int32_t flags)  { @@ -2714,6 +2748,7 @@ struct xlator_fops fops = {          .setattr     = client_setattr,          .fsetattr    = client_fsetattr,  	.fallocate   = client_fallocate, +	.discard     = client_discard,          .getspec     = client_getspec,  }; diff --git a/xlators/protocol/server/src/server-rpc-fops.c b/xlators/protocol/server/src/server-rpc-fops.c index 4b847eabd86..55e30fdd782 100644 --- a/xlators/protocol/server/src/server-rpc-fops.c +++ b/xlators/protocol/server/src/server-rpc-fops.c @@ -2008,6 +2008,45 @@ out:          return 0;  } +int +server_discard_cbk(call_frame_t *frame, void *cookie, xlator_t *this, +                   int32_t op_ret, int32_t op_errno, +                   struct iatt *statpre, struct iatt *statpost, dict_t *xdata) +{ +        gfs3_discard_rsp rsp	 = {0,}; +        server_state_t    *state = NULL; +        rpcsvc_request_t  *req   = NULL; + +        req = frame->local; +        state  = CALL_STATE (frame); + +        GF_PROTOCOL_DICT_SERIALIZE (this, xdata, (&rsp.xdata.xdata_val), +                                    rsp.xdata.xdata_len, op_errno, out); + +        if (op_ret) { +                gf_log (this->name, GF_LOG_INFO, +                        "%"PRId64": DISCARD %"PRId64" (%s) ==> (%s)", +                        frame->root->unique, state->resolve.fd_no, +                        uuid_utoa (state->resolve.gfid), +                        strerror (op_errno)); +                goto out; +        } + +        gf_stat_from_iatt (&rsp.statpre, statpre); +        gf_stat_from_iatt (&rsp.statpost, statpost); + +out: +        rsp.op_ret    = op_ret; +        rsp.op_errno  = gf_errno_to_error (op_errno); + +        server_submit_reply(frame, req, &rsp, NULL, 0, NULL, +                            (xdrproc_t) xdr_gfs3_discard_rsp); + +        GF_FREE (rsp.xdata.xdata_val); + +        return 0; +} +  /* Resume function section */  int @@ -2975,6 +3014,26 @@ err:          return 0;  } +int +server_discard_resume (call_frame_t *frame, xlator_t *bound_xl) +{ +        server_state_t *state = NULL; + +        state = CALL_STATE (frame); + +        if (state->resolve.op_ret != 0) +                goto err; + +        STACK_WIND (frame, server_discard_cbk, +                    bound_xl, bound_xl->fops->discard, +                    state->fd, state->offset, state->size, state->xdata); +        return 0; +err: +        server_discard_cbk(frame, NULL, frame->this, state->resolve.op_ret, +			   state->resolve.op_errno, NULL, NULL, NULL); + +        return 0; +}  /* Fop section */ @@ -3213,6 +3272,68 @@ out:          return ret;  } + + +int +server3_3_discard(rpcsvc_request_t *req) +{ +        server_state_t    *state = NULL; +        call_frame_t      *frame = NULL; +        gfs3_discard_req args	 = {{0},}; +        int                ret   = -1; +        int                op_errno = 0; + +        if (!req) +                return ret; + +        ret = xdr_to_generic (req->msg[0], &args, +                              (xdrproc_t)xdr_gfs3_discard_req); +        if (ret < 0) { +                //failed to decode msg; +                req->rpc_err = GARBAGE_ARGS; +                goto out; +        } + +        frame = get_frame_from_request (req); +        if (!frame) { +                // something wrong, mostly insufficient memory +                req->rpc_err = GARBAGE_ARGS; /* TODO */ +                goto out; +        } +        frame->root->op = GF_FOP_DISCARD; + +        state = CALL_STATE (frame); +        if (!state->conn->bound_xl) { +                /* auth failure, request on subvolume without setvolume */ +                req->rpc_err = GARBAGE_ARGS; +                goto out; +        } + +        state->resolve.type   = RESOLVE_MUST; +        state->resolve.fd_no  = args.fd; + +	state->offset = args.offset; +	state->size = args.size; +	memcpy(state->resolve.gfid, args.gfid, 16); + +        GF_PROTOCOL_DICT_UNSERIALIZE (state->conn->bound_xl, state->xdata, +                                      (args.xdata.xdata_val), +                                      (args.xdata.xdata_len), ret, +                                      op_errno, out); + +        ret = 0; +        resolve_and_resume (frame, server_discard_resume); + +out: +        free (args.xdata.xdata_val); + +        if (op_errno) +                req->rpc_err = GARBAGE_ARGS; + +        return ret; +} + +  int  server3_3_readlink (rpcsvc_request_t *req)  { @@ -5867,6 +5988,7 @@ rpcsvc_actor_t glusterfs3_3_fop_actors[] = {          [GFS3_OP_FSETATTR]    = { "FSETATTR",   GFS3_OP_FSETATTR, server3_3_fsetattr, NULL, 0},          [GFS3_OP_READDIRP]    = { "READDIRP",   GFS3_OP_READDIRP, server3_3_readdirp, NULL, 0},  	[GFS3_OP_FALLOCATE]    = { "FALLOCATE",	GFS3_OP_FALLOCATE, server3_3_fallocate, NULL, 0}, +	[GFS3_OP_DISCARD]     = { "DISCARD",	GFS3_OP_DISCARD, server3_3_discard, NULL, 0},          [GFS3_OP_RELEASE]     = { "RELEASE",    GFS3_OP_RELEASE, server3_3_release, NULL, 0},          [GFS3_OP_RELEASEDIR]  = { "RELEASEDIR", GFS3_OP_RELEASEDIR, server3_3_releasedir, NULL, 0},          [GFS3_OP_FREMOVEXATTR] = { "FREMOVEXATTR", GFS3_OP_FREMOVEXATTR, server3_3_fremovexattr, NULL, 0}, diff --git a/xlators/storage/posix/src/posix.c b/xlators/storage/posix/src/posix.c index 36883403790..05101fd3177 100644 --- a/xlators/storage/posix/src/posix.c +++ b/xlators/storage/posix/src/posix.c @@ -562,16 +562,12 @@ out:  }  static int32_t -_posix_fallocate(call_frame_t *frame, xlator_t *this, fd_t *fd, int32_t keep_size, -		off_t offset, size_t len, dict_t *xdata) +posix_do_fallocate(call_frame_t *frame, xlator_t *this, fd_t *fd, int32_t flags, +		   off_t offset, size_t len, struct iatt *statpre, +		   struct iatt *statpost)  { -	int32_t        op_ret    = -1; -        int32_t        op_errno  = 0; -        struct iatt    statpre     = {0,}; -        struct iatt    statpost    = {0,};          struct posix_fd *pfd = NULL;          int32_t          ret = -1; -	int32_t		flags = 0;          DECLARE_OLD_FS_ID_VAR; @@ -583,48 +579,87 @@ _posix_fallocate(call_frame_t *frame, xlator_t *this, fd_t *fd, int32_t keep_siz          ret = posix_fd_ctx_get (fd, this, &pfd);          if (ret < 0) { -                op_errno = -ret;                  gf_log (this->name, GF_LOG_DEBUG,                          "pfd is NULL from fd=%p", fd);                  goto out;          } -        op_ret = posix_fdstat (this, pfd->fd, &statpre); -        if (op_ret == -1) { -                op_errno = errno; +        ret = posix_fdstat (this, pfd->fd, statpre); +        if (ret == -1) { +                ret = -errno;                  gf_log (this->name, GF_LOG_ERROR,                          "fallocate (fstat) failed on fd=%p: %s", fd, -                        strerror (op_errno)); +                        strerror (errno));                  goto out;          } -	if (keep_size) -		flags = FALLOC_FL_KEEP_SIZE; - -	op_ret = sys_fallocate(pfd->fd, flags, offset, len); -	if (op_ret == -1) { -		op_errno = errno; +	ret = sys_fallocate(pfd->fd, flags, offset, len); +	if (ret == -1) { +		ret = -errno;  		goto out;  	} -        op_ret = posix_fdstat (this, pfd->fd, &statpost); -        if (op_ret == -1) { -                op_errno = errno; +        ret = posix_fdstat (this, pfd->fd, statpost); +        if (ret == -1) { +                ret = -errno;                  gf_log (this->name, GF_LOG_ERROR,                          "fallocate (fstat) failed on fd=%p: %s", fd, -                        strerror (op_errno)); +                        strerror (errno));                  goto out;          } -        op_ret = 0; -  out:          SET_TO_OLD_FS_ID (); -        STACK_UNWIND_STRICT (fallocate, frame, op_ret, op_errno, -                             &statpre, &statpost, NULL); +        return ret; +} + +static int32_t +_posix_fallocate(call_frame_t *frame, xlator_t *this, fd_t *fd, int32_t keep_size, +		off_t offset, size_t len, dict_t *xdata) +{ +	int32_t ret; +	int32_t flags = 0; +        struct iatt statpre = {0,}; +        struct iatt statpost = {0,}; + +	if (keep_size) +		flags = FALLOC_FL_KEEP_SIZE; + +	ret = posix_do_fallocate(frame, this, fd, flags, offset, len, +				 &statpre, &statpost); +	if (ret < 0)  +		goto err; + +	STACK_UNWIND_STRICT(fallocate, frame, 0, 0, &statpre, &statpost, NULL); +	return 0; + +err: +	STACK_UNWIND_STRICT(fallocate, frame, -1, -ret, NULL, NULL, NULL); +	return 0; +} + +static int32_t +posix_discard(call_frame_t *frame, xlator_t *this, fd_t *fd, off_t offset, +	      size_t len, dict_t *xdata) +{ +	int32_t ret; +	int32_t flags = FALLOC_FL_KEEP_SIZE|FALLOC_FL_PUNCH_HOLE; +        struct iatt statpre = {0,}; +        struct iatt statpost = {0,}; + +	ret = posix_do_fallocate(frame, this, fd, flags, offset, len, +				 &statpre, &statpost); +	if (ret < 0)  +		goto err; + +	STACK_UNWIND_STRICT(discard, frame, 0, 0, &statpre, &statpost, NULL); +	return 0; + +err: +	STACK_UNWIND_STRICT(discard, frame, -1, -ret, NULL, NULL, NULL); +	return 0; -        return 0;  }  int32_t @@ -4726,6 +4761,7 @@ struct xlator_fops fops = {          .setattr     = posix_setattr,          .fsetattr    = posix_fsetattr,  	.fallocate   = _posix_fallocate, +	.discard     = posix_discard,  };  struct xlator_cbks cbks = {  | 
