aboutsummaryrefslogtreecommitdiffstats
path: root/fs/ocfs2
diff options
context:
space:
mode:
Diffstat (limited to 'fs/ocfs2')
-rw-r--r--fs/ocfs2/aops.c108
1 files changed, 76 insertions, 32 deletions
diff --git a/fs/ocfs2/aops.c b/fs/ocfs2/aops.c
index 510bf84c9cf5..077583b50391 100644
--- a/fs/ocfs2/aops.c
+++ b/fs/ocfs2/aops.c
@@ -1188,6 +1188,31 @@ out:
1188 return ret; 1188 return ret;
1189} 1189}
1190 1190
1191static int ocfs2_write_cluster_by_desc(struct address_space *mapping,
1192 struct ocfs2_alloc_context *data_ac,
1193 struct ocfs2_alloc_context *meta_ac,
1194 struct ocfs2_write_ctxt *wc,
1195 loff_t pos, unsigned len)
1196{
1197 int ret, i;
1198 struct ocfs2_write_cluster_desc *desc;
1199
1200 for (i = 0; i < wc->w_clen; i++) {
1201 desc = &wc->w_desc[i];
1202
1203 ret = ocfs2_write_cluster(mapping, desc->c_phys, data_ac,
1204 meta_ac, wc, desc->c_cpos, pos, len);
1205 if (ret) {
1206 mlog_errno(ret);
1207 goto out;
1208 }
1209 }
1210
1211 ret = 0;
1212out:
1213 return ret;
1214}
1215
1191/* 1216/*
1192 * ocfs2_write_end() wants to know which parts of the target page it 1217 * ocfs2_write_end() wants to know which parts of the target page it
1193 * should complete the write on. It's easiest to compute them ahead of 1218 * should complete the write on. It's easiest to compute them ahead of
@@ -1240,30 +1265,19 @@ static void ocfs2_set_target_boundaries(struct ocfs2_super *osb,
1240 } 1265 }
1241} 1266}
1242 1267
1243int ocfs2_write_begin_nolock(struct address_space *mapping, 1268/*
1244 loff_t pos, unsigned len, unsigned flags, 1269 * Populate each single-cluster write descriptor in the write context
1245 struct page **pagep, void **fsdata, 1270 * with information about the i/o to be done.
1246 struct buffer_head *di_bh, struct page *mmap_page) 1271 */
1272static int ocfs2_populate_write_desc(struct inode *inode,
1273 struct ocfs2_write_ctxt *wc,
1274 unsigned int *clusters_to_alloc)
1247{ 1275{
1248 int ret, i, credits = OCFS2_INODE_UPDATE_CREDITS; 1276 int ret;
1249 unsigned int num_clusters = 0, clusters_to_alloc = 0;
1250 u32 phys = 0;
1251 struct ocfs2_write_ctxt *wc;
1252 struct inode *inode = mapping->host;
1253 struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
1254 struct ocfs2_dinode *di;
1255 struct ocfs2_alloc_context *data_ac = NULL;
1256 struct ocfs2_alloc_context *meta_ac = NULL;
1257 handle_t *handle;
1258 struct ocfs2_write_cluster_desc *desc; 1277 struct ocfs2_write_cluster_desc *desc;
1259 1278 unsigned int num_clusters = 0;
1260 ret = ocfs2_alloc_write_ctxt(&wc, osb, pos, len, di_bh); 1279 u32 phys = 0;
1261 if (ret) { 1280 int i;
1262 mlog_errno(ret);
1263 return ret;
1264 }
1265
1266 di = (struct ocfs2_dinode *)wc->w_di_bh->b_data;
1267 1281
1268 for (i = 0; i < wc->w_clen; i++) { 1282 for (i = 0; i < wc->w_clen; i++) {
1269 desc = &wc->w_desc[i]; 1283 desc = &wc->w_desc[i];
@@ -1287,12 +1301,46 @@ int ocfs2_write_begin_nolock(struct address_space *mapping,
1287 desc->c_phys = phys; 1301 desc->c_phys = phys;
1288 if (phys == 0) { 1302 if (phys == 0) {
1289 desc->c_new = 1; 1303 desc->c_new = 1;
1290 clusters_to_alloc++; 1304 *clusters_to_alloc = *clusters_to_alloc + 1;
1291 } 1305 }
1292 1306
1293 num_clusters--; 1307 num_clusters--;
1294 } 1308 }
1295 1309
1310 ret = 0;
1311out:
1312 return ret;
1313}
1314
1315int ocfs2_write_begin_nolock(struct address_space *mapping,
1316 loff_t pos, unsigned len, unsigned flags,
1317 struct page **pagep, void **fsdata,
1318 struct buffer_head *di_bh, struct page *mmap_page)
1319{
1320 int ret, credits = OCFS2_INODE_UPDATE_CREDITS;
1321 unsigned int clusters_to_alloc = 0;
1322 struct ocfs2_write_ctxt *wc;
1323 struct inode *inode = mapping->host;
1324 struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
1325 struct ocfs2_dinode *di;
1326 struct ocfs2_alloc_context *data_ac = NULL;
1327 struct ocfs2_alloc_context *meta_ac = NULL;
1328 handle_t *handle;
1329
1330 ret = ocfs2_alloc_write_ctxt(&wc, osb, pos, len, di_bh);
1331 if (ret) {
1332 mlog_errno(ret);
1333 return ret;
1334 }
1335
1336 ret = ocfs2_populate_write_desc(inode, wc, &clusters_to_alloc);
1337 if (ret) {
1338 mlog_errno(ret);
1339 goto out;
1340 }
1341
1342 di = (struct ocfs2_dinode *)wc->w_di_bh->b_data;
1343
1296 /* 1344 /*
1297 * We set w_target_from, w_target_to here so that 1345 * We set w_target_from, w_target_to here so that
1298 * ocfs2_write_end() knows which range in the target page to 1346 * ocfs2_write_end() knows which range in the target page to
@@ -1351,15 +1399,11 @@ int ocfs2_write_begin_nolock(struct address_space *mapping,
1351 goto out_commit; 1399 goto out_commit;
1352 } 1400 }
1353 1401
1354 for (i = 0; i < wc->w_clen; i++) { 1402 ret = ocfs2_write_cluster_by_desc(mapping, data_ac, meta_ac, wc, pos,
1355 desc = &wc->w_desc[i]; 1403 len);
1356 1404 if (ret) {
1357 ret = ocfs2_write_cluster(mapping, desc->c_phys, data_ac, 1405 mlog_errno(ret);
1358 meta_ac, wc, desc->c_cpos, pos, len); 1406 goto out_commit;
1359 if (ret) {
1360 mlog_errno(ret);
1361 goto out_commit;
1362 }
1363 } 1407 }
1364 1408
1365 if (data_ac) 1409 if (data_ac)