diff options
Diffstat (limited to 'fs/ceph/caps.c')
-rw-r--r-- | fs/ceph/caps.c | 42 |
1 files changed, 32 insertions, 10 deletions
diff --git a/fs/ceph/caps.c b/fs/ceph/caps.c index 7d0a0d0adc18..b6fdf010749b 100644 --- a/fs/ceph/caps.c +++ b/fs/ceph/caps.c | |||
@@ -1204,6 +1204,12 @@ retry: | |||
1204 | if (capsnap->dirty_pages || capsnap->writing) | 1204 | if (capsnap->dirty_pages || capsnap->writing) |
1205 | continue; | 1205 | continue; |
1206 | 1206 | ||
1207 | /* | ||
1208 | * if cap writeback already occurred, we should have dropped | ||
1209 | * the capsnap in ceph_put_wrbuffer_cap_refs. | ||
1210 | */ | ||
1211 | BUG_ON(capsnap->dirty == 0); | ||
1212 | |||
1207 | /* pick mds, take s_mutex */ | 1213 | /* pick mds, take s_mutex */ |
1208 | mds = __ceph_get_cap_mds(ci, &mseq); | 1214 | mds = __ceph_get_cap_mds(ci, &mseq); |
1209 | if (session && session->s_mds != mds) { | 1215 | if (session && session->s_mds != mds) { |
@@ -2117,8 +2123,8 @@ void ceph_put_cap_refs(struct ceph_inode_info *ci, int had) | |||
2117 | } | 2123 | } |
2118 | spin_unlock(&inode->i_lock); | 2124 | spin_unlock(&inode->i_lock); |
2119 | 2125 | ||
2120 | dout("put_cap_refs %p had %s %s\n", inode, ceph_cap_string(had), | 2126 | dout("put_cap_refs %p had %s%s%s\n", inode, ceph_cap_string(had), |
2121 | last ? "last" : ""); | 2127 | last ? " last" : "", put ? " put" : ""); |
2122 | 2128 | ||
2123 | if (last && !flushsnaps) | 2129 | if (last && !flushsnaps) |
2124 | ceph_check_caps(ci, 0, NULL); | 2130 | ceph_check_caps(ci, 0, NULL); |
@@ -2142,7 +2148,8 @@ void ceph_put_wrbuffer_cap_refs(struct ceph_inode_info *ci, int nr, | |||
2142 | { | 2148 | { |
2143 | struct inode *inode = &ci->vfs_inode; | 2149 | struct inode *inode = &ci->vfs_inode; |
2144 | int last = 0; | 2150 | int last = 0; |
2145 | int last_snap = 0; | 2151 | int complete_capsnap = 0; |
2152 | int drop_capsnap = 0; | ||
2146 | int found = 0; | 2153 | int found = 0; |
2147 | struct ceph_cap_snap *capsnap = NULL; | 2154 | struct ceph_cap_snap *capsnap = NULL; |
2148 | 2155 | ||
@@ -2165,19 +2172,32 @@ void ceph_put_wrbuffer_cap_refs(struct ceph_inode_info *ci, int nr, | |||
2165 | list_for_each_entry(capsnap, &ci->i_cap_snaps, ci_item) { | 2172 | list_for_each_entry(capsnap, &ci->i_cap_snaps, ci_item) { |
2166 | if (capsnap->context == snapc) { | 2173 | if (capsnap->context == snapc) { |
2167 | found = 1; | 2174 | found = 1; |
2168 | capsnap->dirty_pages -= nr; | ||
2169 | last_snap = !capsnap->dirty_pages; | ||
2170 | break; | 2175 | break; |
2171 | } | 2176 | } |
2172 | } | 2177 | } |
2173 | BUG_ON(!found); | 2178 | BUG_ON(!found); |
2179 | capsnap->dirty_pages -= nr; | ||
2180 | if (capsnap->dirty_pages == 0) { | ||
2181 | complete_capsnap = 1; | ||
2182 | if (capsnap->dirty == 0) | ||
2183 | /* cap writeback completed before we created | ||
2184 | * the cap_snap; no FLUSHSNAP is needed */ | ||
2185 | drop_capsnap = 1; | ||
2186 | } | ||
2174 | dout("put_wrbuffer_cap_refs on %p cap_snap %p " | 2187 | dout("put_wrbuffer_cap_refs on %p cap_snap %p " |
2175 | " snap %lld %d/%d -> %d/%d %s%s\n", | 2188 | " snap %lld %d/%d -> %d/%d %s%s%s\n", |
2176 | inode, capsnap, capsnap->context->seq, | 2189 | inode, capsnap, capsnap->context->seq, |
2177 | ci->i_wrbuffer_ref+nr, capsnap->dirty_pages + nr, | 2190 | ci->i_wrbuffer_ref+nr, capsnap->dirty_pages + nr, |
2178 | ci->i_wrbuffer_ref, capsnap->dirty_pages, | 2191 | ci->i_wrbuffer_ref, capsnap->dirty_pages, |
2179 | last ? " (wrbuffer last)" : "", | 2192 | last ? " (wrbuffer last)" : "", |
2180 | last_snap ? " (capsnap last)" : ""); | 2193 | complete_capsnap ? " (complete capsnap)" : "", |
2194 | drop_capsnap ? " (drop capsnap)" : ""); | ||
2195 | if (drop_capsnap) { | ||
2196 | ceph_put_snap_context(capsnap->context); | ||
2197 | list_del(&capsnap->ci_item); | ||
2198 | list_del(&capsnap->flushing_item); | ||
2199 | ceph_put_cap_snap(capsnap); | ||
2200 | } | ||
2181 | } | 2201 | } |
2182 | 2202 | ||
2183 | spin_unlock(&inode->i_lock); | 2203 | spin_unlock(&inode->i_lock); |
@@ -2185,10 +2205,12 @@ void ceph_put_wrbuffer_cap_refs(struct ceph_inode_info *ci, int nr, | |||
2185 | if (last) { | 2205 | if (last) { |
2186 | ceph_check_caps(ci, CHECK_CAPS_AUTHONLY, NULL); | 2206 | ceph_check_caps(ci, CHECK_CAPS_AUTHONLY, NULL); |
2187 | iput(inode); | 2207 | iput(inode); |
2188 | } else if (last_snap) { | 2208 | } else if (complete_capsnap) { |
2189 | ceph_flush_snaps(ci); | 2209 | ceph_flush_snaps(ci); |
2190 | wake_up(&ci->i_cap_wq); | 2210 | wake_up(&ci->i_cap_wq); |
2191 | } | 2211 | } |
2212 | if (drop_capsnap) | ||
2213 | iput(inode); | ||
2192 | } | 2214 | } |
2193 | 2215 | ||
2194 | /* | 2216 | /* |
@@ -2464,8 +2486,8 @@ static void handle_cap_flushsnap_ack(struct inode *inode, u64 flush_tid, | |||
2464 | break; | 2486 | break; |
2465 | } | 2487 | } |
2466 | WARN_ON(capsnap->dirty_pages || capsnap->writing); | 2488 | WARN_ON(capsnap->dirty_pages || capsnap->writing); |
2467 | dout(" removing cap_snap %p follows %lld\n", | 2489 | dout(" removing %p cap_snap %p follows %lld\n", |
2468 | capsnap, follows); | 2490 | inode, capsnap, follows); |
2469 | ceph_put_snap_context(capsnap->context); | 2491 | ceph_put_snap_context(capsnap->context); |
2470 | list_del(&capsnap->ci_item); | 2492 | list_del(&capsnap->ci_item); |
2471 | list_del(&capsnap->flushing_item); | 2493 | list_del(&capsnap->flushing_item); |