aboutsummaryrefslogtreecommitdiffstats
path: root/mm/memcontrol.c
diff options
context:
space:
mode:
Diffstat (limited to 'mm/memcontrol.c')
-rw-r--r--mm/memcontrol.c74
1 files changed, 62 insertions, 12 deletions
diff --git a/mm/memcontrol.c b/mm/memcontrol.c
index 81b0ae8183d0..55dea5968464 100644
--- a/mm/memcontrol.c
+++ b/mm/memcontrol.c
@@ -991,10 +991,31 @@ nomem:
991 return -ENOMEM; 991 return -ENOMEM;
992} 992}
993 993
994
995/*
996 * A helper function to get mem_cgroup from ID. must be called under
997 * rcu_read_lock(). The caller must check css_is_removed() or some if
998 * it's concern. (dropping refcnt from swap can be called against removed
999 * memcg.)
1000 */
1001static struct mem_cgroup *mem_cgroup_lookup(unsigned short id)
1002{
1003 struct cgroup_subsys_state *css;
1004
1005 /* ID 0 is unused ID */
1006 if (!id)
1007 return NULL;
1008 css = css_lookup(&mem_cgroup_subsys, id);
1009 if (!css)
1010 return NULL;
1011 return container_of(css, struct mem_cgroup, css);
1012}
1013
994static struct mem_cgroup *try_get_mem_cgroup_from_swapcache(struct page *page) 1014static struct mem_cgroup *try_get_mem_cgroup_from_swapcache(struct page *page)
995{ 1015{
996 struct mem_cgroup *mem; 1016 struct mem_cgroup *mem;
997 struct page_cgroup *pc; 1017 struct page_cgroup *pc;
1018 unsigned short id;
998 swp_entry_t ent; 1019 swp_entry_t ent;
999 1020
1000 VM_BUG_ON(!PageLocked(page)); 1021 VM_BUG_ON(!PageLocked(page));
@@ -1006,16 +1027,19 @@ static struct mem_cgroup *try_get_mem_cgroup_from_swapcache(struct page *page)
1006 /* 1027 /*
1007 * Used bit of swapcache is solid under page lock. 1028 * Used bit of swapcache is solid under page lock.
1008 */ 1029 */
1009 if (PageCgroupUsed(pc)) 1030 if (PageCgroupUsed(pc)) {
1010 mem = pc->mem_cgroup; 1031 mem = pc->mem_cgroup;
1011 else { 1032 if (mem && !css_tryget(&mem->css))
1033 mem = NULL;
1034 } else {
1012 ent.val = page_private(page); 1035 ent.val = page_private(page);
1013 mem = lookup_swap_cgroup(ent); 1036 id = lookup_swap_cgroup(ent);
1037 rcu_read_lock();
1038 mem = mem_cgroup_lookup(id);
1039 if (mem && !css_tryget(&mem->css))
1040 mem = NULL;
1041 rcu_read_unlock();
1014 } 1042 }
1015 if (!mem)
1016 return NULL;
1017 if (!css_tryget(&mem->css))
1018 return NULL;
1019 return mem; 1043 return mem;
1020} 1044}
1021 1045
@@ -1276,12 +1300,22 @@ int mem_cgroup_cache_charge(struct page *page, struct mm_struct *mm,
1276 1300
1277 if (do_swap_account && !ret && PageSwapCache(page)) { 1301 if (do_swap_account && !ret && PageSwapCache(page)) {
1278 swp_entry_t ent = {.val = page_private(page)}; 1302 swp_entry_t ent = {.val = page_private(page)};
1303 unsigned short id;
1279 /* avoid double counting */ 1304 /* avoid double counting */
1280 mem = swap_cgroup_record(ent, NULL); 1305 id = swap_cgroup_record(ent, 0);
1306 rcu_read_lock();
1307 mem = mem_cgroup_lookup(id);
1281 if (mem) { 1308 if (mem) {
1309 /*
1310 * We did swap-in. Then, this entry is doubly counted
1311 * both in mem and memsw. We uncharge it, here.
1312 * Recorded ID can be obsolete. We avoid calling
1313 * css_tryget()
1314 */
1282 res_counter_uncharge(&mem->memsw, PAGE_SIZE); 1315 res_counter_uncharge(&mem->memsw, PAGE_SIZE);
1283 mem_cgroup_put(mem); 1316 mem_cgroup_put(mem);
1284 } 1317 }
1318 rcu_read_unlock();
1285 } 1319 }
1286 return ret; 1320 return ret;
1287} 1321}
@@ -1346,13 +1380,21 @@ void mem_cgroup_commit_charge_swapin(struct page *page, struct mem_cgroup *ptr)
1346 */ 1380 */
1347 if (do_swap_account && PageSwapCache(page)) { 1381 if (do_swap_account && PageSwapCache(page)) {
1348 swp_entry_t ent = {.val = page_private(page)}; 1382 swp_entry_t ent = {.val = page_private(page)};
1383 unsigned short id;
1349 struct mem_cgroup *memcg; 1384 struct mem_cgroup *memcg;
1350 memcg = swap_cgroup_record(ent, NULL); 1385
1386 id = swap_cgroup_record(ent, 0);
1387 rcu_read_lock();
1388 memcg = mem_cgroup_lookup(id);
1351 if (memcg) { 1389 if (memcg) {
1390 /*
1391 * This recorded memcg can be obsolete one. So, avoid
1392 * calling css_tryget
1393 */
1352 res_counter_uncharge(&memcg->memsw, PAGE_SIZE); 1394 res_counter_uncharge(&memcg->memsw, PAGE_SIZE);
1353 mem_cgroup_put(memcg); 1395 mem_cgroup_put(memcg);
1354 } 1396 }
1355 1397 rcu_read_unlock();
1356 } 1398 }
1357 /* add this page(page_cgroup) to the LRU we want. */ 1399 /* add this page(page_cgroup) to the LRU we want. */
1358 1400
@@ -1473,7 +1515,7 @@ void mem_cgroup_uncharge_swapcache(struct page *page, swp_entry_t ent)
1473 MEM_CGROUP_CHARGE_TYPE_SWAPOUT); 1515 MEM_CGROUP_CHARGE_TYPE_SWAPOUT);
1474 /* record memcg information */ 1516 /* record memcg information */
1475 if (do_swap_account && memcg) { 1517 if (do_swap_account && memcg) {
1476 swap_cgroup_record(ent, memcg); 1518 swap_cgroup_record(ent, css_id(&memcg->css));
1477 mem_cgroup_get(memcg); 1519 mem_cgroup_get(memcg);
1478 } 1520 }
1479 if (memcg) 1521 if (memcg)
@@ -1488,15 +1530,23 @@ void mem_cgroup_uncharge_swapcache(struct page *page, swp_entry_t ent)
1488void mem_cgroup_uncharge_swap(swp_entry_t ent) 1530void mem_cgroup_uncharge_swap(swp_entry_t ent)
1489{ 1531{
1490 struct mem_cgroup *memcg; 1532 struct mem_cgroup *memcg;
1533 unsigned short id;
1491 1534
1492 if (!do_swap_account) 1535 if (!do_swap_account)
1493 return; 1536 return;
1494 1537
1495 memcg = swap_cgroup_record(ent, NULL); 1538 id = swap_cgroup_record(ent, 0);
1539 rcu_read_lock();
1540 memcg = mem_cgroup_lookup(id);
1496 if (memcg) { 1541 if (memcg) {
1542 /*
1543 * We uncharge this because swap is freed.
1544 * This memcg can be obsolete one. We avoid calling css_tryget
1545 */
1497 res_counter_uncharge(&memcg->memsw, PAGE_SIZE); 1546 res_counter_uncharge(&memcg->memsw, PAGE_SIZE);
1498 mem_cgroup_put(memcg); 1547 mem_cgroup_put(memcg);
1499 } 1548 }
1549 rcu_read_unlock();
1500} 1550}
1501#endif 1551#endif
1502 1552