summaryrefslogtreecommitdiffstats
path: root/xlators/features/metadisp/src/metadisp-stat.c
blob: b06d0dbcddd326ae29ef630b49296f052aa6a87d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
#include "metadisp.h"
#include <glusterfs/call-stub.h>

/**
 * The stat flow in METADISP is complicated because we must
 * do ensure a few things:
 *    1. stat, on the path within the metadata layer,
 *       MUST get the backend FD of the data layer.
 *        --- we wind to the metadata layer, then the data layer.
 *
 *    2. the metadata layer MUST be able to ask the data
 *       layer for stat information.
 *        --- this is 'syncop-internal-from-posix'
 *
 *    3. when the metadata exists BUT the data is missing,
 *       we MUST mark the backend file as bad and heal it.
 */

int32_t
metadisp_stat_backend_cbk(call_frame_t *frame, void *cookie, xlator_t *this,
                          int32_t op_ret, int32_t op_errno, struct iatt *buf,
                          dict_t *xdata)
{
    METADISP_TRACE("got backend stat results %d %d", op_ret, op_errno);
    if (op_errno == ENOENT) {
        STACK_UNWIND_STRICT(open, frame, -1, ENODATA, NULL, NULL);
        return 0;
    }
    STACK_UNWIND_STRICT(stat, frame, op_ret, op_errno, buf, xdata);
    return 0;
}

int32_t
metadisp_stat_resume(call_frame_t *frame, xlator_t *this, loc_t *loc,
                     dict_t *xdata)
{
    METADISP_TRACE("winding stat to path %s", loc->path);
    if (gf_uuid_is_null(loc->gfid)) {
        METADISP_TRACE("bad object, sending EUCLEAN");
        STACK_UNWIND_STRICT(open, frame, -1, EUCLEAN, NULL, NULL);
        return 0;
    }

    STACK_WIND(frame, metadisp_stat_backend_cbk, SECOND_CHILD(this),
               SECOND_CHILD(this)->fops->stat, loc, xdata);
    return 0;
}

int32_t
metadisp_stat_cbk(call_frame_t *frame, void *cookie, xlator_t *this,
                  int32_t op_ret, int32_t op_errno, struct iatt *buf,
                  dict_t *xdata)
{
    call_stub_t *stub = NULL;

    METADISP_TRACE("got stat results %d %d", op_ret, op_errno);

    if (cookie) {
        stub = cookie;
    }

    if (op_ret != 0) {
        goto unwind;
    }

    // only use the stub for the files
    if (!IA_ISREG(buf->ia_type)) {
        goto unwind;
    }

    if (stub->poison) {
        call_stub_destroy(stub);
        stub = NULL;
        return 0;
    }

    call_resume(stub);
    return 0;

unwind:
    if (stub) {
        call_stub_destroy(stub);
    }
    STACK_UNWIND_STRICT(stat, frame, op_ret, op_errno, buf, xdata);
    return 0;
}

int32_t
metadisp_stat(call_frame_t *frame, xlator_t *this, loc_t *loc, dict_t *xdata)
{
    call_stub_t *stub = NULL;
    int32_t ret = 0;
    loc_t backend_loc = {
        0,
    };
    METADISP_FILTER_ROOT(stat, loc, xdata);

    if (build_backend_loc(loc->gfid, loc, &backend_loc)) {
        goto unwind;
    }

    if (dict_get_int32(xdata, "syncop-internal-from-posix", &ret) == 0) {
        // if we've just been sent a stat from posix, then we know
        // that we must send down a stat for a file to the second child.
        //
        // that means we can skip the stat for the first child and just
        // send to the data disk.
        METADISP_TRACE("got syncop-internal-from-posix");
        STACK_WIND(frame, default_stat_cbk, DATA_CHILD(this),
                   DATA_CHILD(this)->fops->stat, &backend_loc, xdata);
        return 0;
    }

    // we do not know if the request is for a file, folder, etc. wind
    // to first child to find out.
    stub = fop_stat_stub(frame, metadisp_stat_resume, &backend_loc, xdata);
    METADISP_TRACE("winding stat to first child %s", loc->path);
    STACK_WIND_COOKIE(frame, metadisp_stat_cbk, stub, METADATA_CHILD(this),
                      METADATA_CHILD(this)->fops->stat, loc, xdata);
    return 0;
unwind:
    STACK_UNWIND_STRICT(stat, frame, -1, EINVAL, NULL, NULL);
    return 0;
}