18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * linux/mm/page_io.c 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright (C) 1991, 1992, 1993, 1994 Linus Torvalds 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * Swap reorganised 29.12.95, 88c2ecf20Sopenharmony_ci * Asynchronous swapping added 30.12.95. Stephen Tweedie 98c2ecf20Sopenharmony_ci * Removed race in async swapping. 14.4.1996. Bruno Haible 108c2ecf20Sopenharmony_ci * Add swap of shared pages through the page cache. 20.2.1998. Stephen Tweedie 118c2ecf20Sopenharmony_ci * Always use brw_page, life becomes simpler. 12 May 1998 Eric Biederman 128c2ecf20Sopenharmony_ci */ 138c2ecf20Sopenharmony_ci 148c2ecf20Sopenharmony_ci#include <linux/mm.h> 158c2ecf20Sopenharmony_ci#include <linux/kernel_stat.h> 168c2ecf20Sopenharmony_ci#include <linux/gfp.h> 178c2ecf20Sopenharmony_ci#include <linux/pagemap.h> 188c2ecf20Sopenharmony_ci#include <linux/swap.h> 198c2ecf20Sopenharmony_ci#include <linux/bio.h> 208c2ecf20Sopenharmony_ci#include <linux/swapops.h> 218c2ecf20Sopenharmony_ci#include <linux/buffer_head.h> 228c2ecf20Sopenharmony_ci#include <linux/writeback.h> 238c2ecf20Sopenharmony_ci#include <linux/frontswap.h> 248c2ecf20Sopenharmony_ci#include <linux/blkdev.h> 258c2ecf20Sopenharmony_ci#include <linux/psi.h> 268c2ecf20Sopenharmony_ci#include <linux/uio.h> 278c2ecf20Sopenharmony_ci#include <linux/sched/task.h> 288c2ecf20Sopenharmony_ci 298c2ecf20Sopenharmony_cistatic struct bio *get_swap_bio(gfp_t gfp_flags, 308c2ecf20Sopenharmony_ci struct page *page, bio_end_io_t end_io) 318c2ecf20Sopenharmony_ci{ 328c2ecf20Sopenharmony_ci struct bio *bio; 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci bio = bio_alloc(gfp_flags, 1); 358c2ecf20Sopenharmony_ci if (bio) { 368c2ecf20Sopenharmony_ci struct block_device *bdev; 378c2ecf20Sopenharmony_ci 388c2ecf20Sopenharmony_ci bio->bi_iter.bi_sector = map_swap_page(page, &bdev); 398c2ecf20Sopenharmony_ci bio_set_dev(bio, bdev); 408c2ecf20Sopenharmony_ci bio->bi_iter.bi_sector <<= PAGE_SHIFT - 9; 418c2ecf20Sopenharmony_ci bio->bi_end_io = end_io; 428c2ecf20Sopenharmony_ci 438c2ecf20Sopenharmony_ci bio_add_page(bio, page, thp_size(page), 0); 448c2ecf20Sopenharmony_ci } 458c2ecf20Sopenharmony_ci return bio; 468c2ecf20Sopenharmony_ci} 478c2ecf20Sopenharmony_ci 488c2ecf20Sopenharmony_civoid end_swap_bio_write(struct bio *bio) 498c2ecf20Sopenharmony_ci{ 508c2ecf20Sopenharmony_ci struct page *page = bio_first_page_all(bio); 518c2ecf20Sopenharmony_ci 528c2ecf20Sopenharmony_ci if (bio->bi_status) { 538c2ecf20Sopenharmony_ci SetPageError(page); 548c2ecf20Sopenharmony_ci /* 558c2ecf20Sopenharmony_ci * We failed to write the page out to swap-space. 568c2ecf20Sopenharmony_ci * Re-dirty the page in order to avoid it being reclaimed. 578c2ecf20Sopenharmony_ci * Also print a dire warning that things will go BAD (tm) 588c2ecf20Sopenharmony_ci * very quickly. 598c2ecf20Sopenharmony_ci * 608c2ecf20Sopenharmony_ci * Also clear PG_reclaim to avoid rotate_reclaimable_page() 618c2ecf20Sopenharmony_ci */ 628c2ecf20Sopenharmony_ci set_page_dirty(page); 638c2ecf20Sopenharmony_ci pr_alert("Write-error on swap-device (%u:%u:%llu)\n", 648c2ecf20Sopenharmony_ci MAJOR(bio_dev(bio)), MINOR(bio_dev(bio)), 658c2ecf20Sopenharmony_ci (unsigned long long)bio->bi_iter.bi_sector); 668c2ecf20Sopenharmony_ci ClearPageReclaim(page); 678c2ecf20Sopenharmony_ci } 688c2ecf20Sopenharmony_ci end_page_writeback(page); 698c2ecf20Sopenharmony_ci bio_put(bio); 708c2ecf20Sopenharmony_ci} 718c2ecf20Sopenharmony_ci 728c2ecf20Sopenharmony_cistatic void end_swap_bio_read(struct bio *bio) 738c2ecf20Sopenharmony_ci{ 748c2ecf20Sopenharmony_ci struct page *page = bio_first_page_all(bio); 758c2ecf20Sopenharmony_ci struct task_struct *waiter = bio->bi_private; 768c2ecf20Sopenharmony_ci 778c2ecf20Sopenharmony_ci if (bio->bi_status) { 788c2ecf20Sopenharmony_ci SetPageError(page); 798c2ecf20Sopenharmony_ci ClearPageUptodate(page); 808c2ecf20Sopenharmony_ci pr_alert("Read-error on swap-device (%u:%u:%llu)\n", 818c2ecf20Sopenharmony_ci MAJOR(bio_dev(bio)), MINOR(bio_dev(bio)), 828c2ecf20Sopenharmony_ci (unsigned long long)bio->bi_iter.bi_sector); 838c2ecf20Sopenharmony_ci goto out; 848c2ecf20Sopenharmony_ci } 858c2ecf20Sopenharmony_ci 868c2ecf20Sopenharmony_ci SetPageUptodate(page); 878c2ecf20Sopenharmony_ciout: 888c2ecf20Sopenharmony_ci unlock_page(page); 898c2ecf20Sopenharmony_ci WRITE_ONCE(bio->bi_private, NULL); 908c2ecf20Sopenharmony_ci bio_put(bio); 918c2ecf20Sopenharmony_ci if (waiter) { 928c2ecf20Sopenharmony_ci blk_wake_io_task(waiter); 938c2ecf20Sopenharmony_ci put_task_struct(waiter); 948c2ecf20Sopenharmony_ci } 958c2ecf20Sopenharmony_ci} 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ciint generic_swapfile_activate(struct swap_info_struct *sis, 988c2ecf20Sopenharmony_ci struct file *swap_file, 998c2ecf20Sopenharmony_ci sector_t *span) 1008c2ecf20Sopenharmony_ci{ 1018c2ecf20Sopenharmony_ci struct address_space *mapping = swap_file->f_mapping; 1028c2ecf20Sopenharmony_ci struct inode *inode = mapping->host; 1038c2ecf20Sopenharmony_ci unsigned blocks_per_page; 1048c2ecf20Sopenharmony_ci unsigned long page_no; 1058c2ecf20Sopenharmony_ci unsigned blkbits; 1068c2ecf20Sopenharmony_ci sector_t probe_block; 1078c2ecf20Sopenharmony_ci sector_t last_block; 1088c2ecf20Sopenharmony_ci sector_t lowest_block = -1; 1098c2ecf20Sopenharmony_ci sector_t highest_block = 0; 1108c2ecf20Sopenharmony_ci int nr_extents = 0; 1118c2ecf20Sopenharmony_ci int ret; 1128c2ecf20Sopenharmony_ci 1138c2ecf20Sopenharmony_ci blkbits = inode->i_blkbits; 1148c2ecf20Sopenharmony_ci blocks_per_page = PAGE_SIZE >> blkbits; 1158c2ecf20Sopenharmony_ci 1168c2ecf20Sopenharmony_ci /* 1178c2ecf20Sopenharmony_ci * Map all the blocks into the extent tree. This code doesn't try 1188c2ecf20Sopenharmony_ci * to be very smart. 1198c2ecf20Sopenharmony_ci */ 1208c2ecf20Sopenharmony_ci probe_block = 0; 1218c2ecf20Sopenharmony_ci page_no = 0; 1228c2ecf20Sopenharmony_ci last_block = i_size_read(inode) >> blkbits; 1238c2ecf20Sopenharmony_ci while ((probe_block + blocks_per_page) <= last_block && 1248c2ecf20Sopenharmony_ci page_no < sis->max) { 1258c2ecf20Sopenharmony_ci unsigned block_in_page; 1268c2ecf20Sopenharmony_ci sector_t first_block; 1278c2ecf20Sopenharmony_ci 1288c2ecf20Sopenharmony_ci cond_resched(); 1298c2ecf20Sopenharmony_ci 1308c2ecf20Sopenharmony_ci first_block = probe_block; 1318c2ecf20Sopenharmony_ci ret = bmap(inode, &first_block); 1328c2ecf20Sopenharmony_ci if (ret || !first_block) 1338c2ecf20Sopenharmony_ci goto bad_bmap; 1348c2ecf20Sopenharmony_ci 1358c2ecf20Sopenharmony_ci /* 1368c2ecf20Sopenharmony_ci * It must be PAGE_SIZE aligned on-disk 1378c2ecf20Sopenharmony_ci */ 1388c2ecf20Sopenharmony_ci if (first_block & (blocks_per_page - 1)) { 1398c2ecf20Sopenharmony_ci probe_block++; 1408c2ecf20Sopenharmony_ci goto reprobe; 1418c2ecf20Sopenharmony_ci } 1428c2ecf20Sopenharmony_ci 1438c2ecf20Sopenharmony_ci for (block_in_page = 1; block_in_page < blocks_per_page; 1448c2ecf20Sopenharmony_ci block_in_page++) { 1458c2ecf20Sopenharmony_ci sector_t block; 1468c2ecf20Sopenharmony_ci 1478c2ecf20Sopenharmony_ci block = probe_block + block_in_page; 1488c2ecf20Sopenharmony_ci ret = bmap(inode, &block); 1498c2ecf20Sopenharmony_ci if (ret || !block) 1508c2ecf20Sopenharmony_ci goto bad_bmap; 1518c2ecf20Sopenharmony_ci 1528c2ecf20Sopenharmony_ci if (block != first_block + block_in_page) { 1538c2ecf20Sopenharmony_ci /* Discontiguity */ 1548c2ecf20Sopenharmony_ci probe_block++; 1558c2ecf20Sopenharmony_ci goto reprobe; 1568c2ecf20Sopenharmony_ci } 1578c2ecf20Sopenharmony_ci } 1588c2ecf20Sopenharmony_ci 1598c2ecf20Sopenharmony_ci first_block >>= (PAGE_SHIFT - blkbits); 1608c2ecf20Sopenharmony_ci if (page_no) { /* exclude the header page */ 1618c2ecf20Sopenharmony_ci if (first_block < lowest_block) 1628c2ecf20Sopenharmony_ci lowest_block = first_block; 1638c2ecf20Sopenharmony_ci if (first_block > highest_block) 1648c2ecf20Sopenharmony_ci highest_block = first_block; 1658c2ecf20Sopenharmony_ci } 1668c2ecf20Sopenharmony_ci 1678c2ecf20Sopenharmony_ci /* 1688c2ecf20Sopenharmony_ci * We found a PAGE_SIZE-length, PAGE_SIZE-aligned run of blocks 1698c2ecf20Sopenharmony_ci */ 1708c2ecf20Sopenharmony_ci ret = add_swap_extent(sis, page_no, 1, first_block); 1718c2ecf20Sopenharmony_ci if (ret < 0) 1728c2ecf20Sopenharmony_ci goto out; 1738c2ecf20Sopenharmony_ci nr_extents += ret; 1748c2ecf20Sopenharmony_ci page_no++; 1758c2ecf20Sopenharmony_ci probe_block += blocks_per_page; 1768c2ecf20Sopenharmony_cireprobe: 1778c2ecf20Sopenharmony_ci continue; 1788c2ecf20Sopenharmony_ci } 1798c2ecf20Sopenharmony_ci ret = nr_extents; 1808c2ecf20Sopenharmony_ci *span = 1 + highest_block - lowest_block; 1818c2ecf20Sopenharmony_ci if (page_no == 0) 1828c2ecf20Sopenharmony_ci page_no = 1; /* force Empty message */ 1838c2ecf20Sopenharmony_ci sis->max = page_no; 1848c2ecf20Sopenharmony_ci sis->pages = page_no - 1; 1858c2ecf20Sopenharmony_ci sis->highest_bit = page_no - 1; 1868c2ecf20Sopenharmony_ciout: 1878c2ecf20Sopenharmony_ci return ret; 1888c2ecf20Sopenharmony_cibad_bmap: 1898c2ecf20Sopenharmony_ci pr_err("swapon: swapfile has holes\n"); 1908c2ecf20Sopenharmony_ci ret = -EINVAL; 1918c2ecf20Sopenharmony_ci goto out; 1928c2ecf20Sopenharmony_ci} 1938c2ecf20Sopenharmony_ci 1948c2ecf20Sopenharmony_ci/* 1958c2ecf20Sopenharmony_ci * We may have stale swap cache pages in memory: notice 1968c2ecf20Sopenharmony_ci * them here and get rid of the unnecessary final write. 1978c2ecf20Sopenharmony_ci */ 1988c2ecf20Sopenharmony_ciint swap_writepage(struct page *page, struct writeback_control *wbc) 1998c2ecf20Sopenharmony_ci{ 2008c2ecf20Sopenharmony_ci int ret = 0; 2018c2ecf20Sopenharmony_ci 2028c2ecf20Sopenharmony_ci if (try_to_free_swap(page)) { 2038c2ecf20Sopenharmony_ci unlock_page(page); 2048c2ecf20Sopenharmony_ci goto out; 2058c2ecf20Sopenharmony_ci } 2068c2ecf20Sopenharmony_ci /* 2078c2ecf20Sopenharmony_ci * Arch code may have to preserve more data than just the page 2088c2ecf20Sopenharmony_ci * contents, e.g. memory tags. 2098c2ecf20Sopenharmony_ci */ 2108c2ecf20Sopenharmony_ci ret = arch_prepare_to_swap(page); 2118c2ecf20Sopenharmony_ci if (ret) { 2128c2ecf20Sopenharmony_ci set_page_dirty(page); 2138c2ecf20Sopenharmony_ci unlock_page(page); 2148c2ecf20Sopenharmony_ci goto out; 2158c2ecf20Sopenharmony_ci } 2168c2ecf20Sopenharmony_ci if (frontswap_store(page) == 0) { 2178c2ecf20Sopenharmony_ci set_page_writeback(page); 2188c2ecf20Sopenharmony_ci unlock_page(page); 2198c2ecf20Sopenharmony_ci end_page_writeback(page); 2208c2ecf20Sopenharmony_ci goto out; 2218c2ecf20Sopenharmony_ci } 2228c2ecf20Sopenharmony_ci ret = __swap_writepage(page, wbc, end_swap_bio_write); 2238c2ecf20Sopenharmony_ciout: 2248c2ecf20Sopenharmony_ci return ret; 2258c2ecf20Sopenharmony_ci} 2268c2ecf20Sopenharmony_ci 2278c2ecf20Sopenharmony_cistatic inline void count_swpout_vm_event(struct page *page) 2288c2ecf20Sopenharmony_ci{ 2298c2ecf20Sopenharmony_ci#ifdef CONFIG_TRANSPARENT_HUGEPAGE 2308c2ecf20Sopenharmony_ci if (unlikely(PageTransHuge(page))) 2318c2ecf20Sopenharmony_ci count_vm_event(THP_SWPOUT); 2328c2ecf20Sopenharmony_ci#endif 2338c2ecf20Sopenharmony_ci count_vm_events(PSWPOUT, thp_nr_pages(page)); 2348c2ecf20Sopenharmony_ci} 2358c2ecf20Sopenharmony_ci 2368c2ecf20Sopenharmony_ci#if defined(CONFIG_MEMCG) && defined(CONFIG_BLK_CGROUP) 2378c2ecf20Sopenharmony_cistatic void bio_associate_blkg_from_page(struct bio *bio, struct page *page) 2388c2ecf20Sopenharmony_ci{ 2398c2ecf20Sopenharmony_ci struct cgroup_subsys_state *css; 2408c2ecf20Sopenharmony_ci 2418c2ecf20Sopenharmony_ci if (!page->mem_cgroup) 2428c2ecf20Sopenharmony_ci return; 2438c2ecf20Sopenharmony_ci 2448c2ecf20Sopenharmony_ci rcu_read_lock(); 2458c2ecf20Sopenharmony_ci css = cgroup_e_css(page->mem_cgroup->css.cgroup, &io_cgrp_subsys); 2468c2ecf20Sopenharmony_ci bio_associate_blkg_from_css(bio, css); 2478c2ecf20Sopenharmony_ci rcu_read_unlock(); 2488c2ecf20Sopenharmony_ci} 2498c2ecf20Sopenharmony_ci#else 2508c2ecf20Sopenharmony_ci#define bio_associate_blkg_from_page(bio, page) do { } while (0) 2518c2ecf20Sopenharmony_ci#endif /* CONFIG_MEMCG && CONFIG_BLK_CGROUP */ 2528c2ecf20Sopenharmony_ci 2538c2ecf20Sopenharmony_ciint __swap_writepage(struct page *page, struct writeback_control *wbc, 2548c2ecf20Sopenharmony_ci bio_end_io_t end_write_func) 2558c2ecf20Sopenharmony_ci{ 2568c2ecf20Sopenharmony_ci struct bio *bio; 2578c2ecf20Sopenharmony_ci int ret; 2588c2ecf20Sopenharmony_ci struct swap_info_struct *sis = page_swap_info(page); 2598c2ecf20Sopenharmony_ci 2608c2ecf20Sopenharmony_ci VM_BUG_ON_PAGE(!PageSwapCache(page), page); 2618c2ecf20Sopenharmony_ci if (data_race(sis->flags & SWP_FS_OPS)) { 2628c2ecf20Sopenharmony_ci struct kiocb kiocb; 2638c2ecf20Sopenharmony_ci struct file *swap_file = sis->swap_file; 2648c2ecf20Sopenharmony_ci struct address_space *mapping = swap_file->f_mapping; 2658c2ecf20Sopenharmony_ci struct bio_vec bv = { 2668c2ecf20Sopenharmony_ci .bv_page = page, 2678c2ecf20Sopenharmony_ci .bv_len = PAGE_SIZE, 2688c2ecf20Sopenharmony_ci .bv_offset = 0 2698c2ecf20Sopenharmony_ci }; 2708c2ecf20Sopenharmony_ci struct iov_iter from; 2718c2ecf20Sopenharmony_ci 2728c2ecf20Sopenharmony_ci iov_iter_bvec(&from, WRITE, &bv, 1, PAGE_SIZE); 2738c2ecf20Sopenharmony_ci init_sync_kiocb(&kiocb, swap_file); 2748c2ecf20Sopenharmony_ci kiocb.ki_pos = page_file_offset(page); 2758c2ecf20Sopenharmony_ci 2768c2ecf20Sopenharmony_ci set_page_writeback(page); 2778c2ecf20Sopenharmony_ci unlock_page(page); 2788c2ecf20Sopenharmony_ci ret = mapping->a_ops->direct_IO(&kiocb, &from); 2798c2ecf20Sopenharmony_ci if (ret == PAGE_SIZE) { 2808c2ecf20Sopenharmony_ci count_vm_event(PSWPOUT); 2818c2ecf20Sopenharmony_ci ret = 0; 2828c2ecf20Sopenharmony_ci } else { 2838c2ecf20Sopenharmony_ci /* 2848c2ecf20Sopenharmony_ci * In the case of swap-over-nfs, this can be a 2858c2ecf20Sopenharmony_ci * temporary failure if the system has limited 2868c2ecf20Sopenharmony_ci * memory for allocating transmit buffers. 2878c2ecf20Sopenharmony_ci * Mark the page dirty and avoid 2888c2ecf20Sopenharmony_ci * rotate_reclaimable_page but rate-limit the 2898c2ecf20Sopenharmony_ci * messages but do not flag PageError like 2908c2ecf20Sopenharmony_ci * the normal direct-to-bio case as it could 2918c2ecf20Sopenharmony_ci * be temporary. 2928c2ecf20Sopenharmony_ci */ 2938c2ecf20Sopenharmony_ci set_page_dirty(page); 2948c2ecf20Sopenharmony_ci ClearPageReclaim(page); 2958c2ecf20Sopenharmony_ci pr_err_ratelimited("Write error on dio swapfile (%llu)\n", 2968c2ecf20Sopenharmony_ci page_file_offset(page)); 2978c2ecf20Sopenharmony_ci } 2988c2ecf20Sopenharmony_ci end_page_writeback(page); 2998c2ecf20Sopenharmony_ci return ret; 3008c2ecf20Sopenharmony_ci } 3018c2ecf20Sopenharmony_ci 3028c2ecf20Sopenharmony_ci ret = bdev_write_page(sis->bdev, swap_page_sector(page), page, wbc); 3038c2ecf20Sopenharmony_ci if (!ret) { 3048c2ecf20Sopenharmony_ci count_swpout_vm_event(page); 3058c2ecf20Sopenharmony_ci return 0; 3068c2ecf20Sopenharmony_ci } 3078c2ecf20Sopenharmony_ci 3088c2ecf20Sopenharmony_ci bio = get_swap_bio(GFP_NOIO, page, end_write_func); 3098c2ecf20Sopenharmony_ci if (bio == NULL) { 3108c2ecf20Sopenharmony_ci set_page_dirty(page); 3118c2ecf20Sopenharmony_ci unlock_page(page); 3128c2ecf20Sopenharmony_ci return -ENOMEM; 3138c2ecf20Sopenharmony_ci } 3148c2ecf20Sopenharmony_ci bio->bi_opf = REQ_OP_WRITE | REQ_SWAP | wbc_to_write_flags(wbc); 3158c2ecf20Sopenharmony_ci bio_associate_blkg_from_page(bio, page); 3168c2ecf20Sopenharmony_ci count_swpout_vm_event(page); 3178c2ecf20Sopenharmony_ci set_page_writeback(page); 3188c2ecf20Sopenharmony_ci unlock_page(page); 3198c2ecf20Sopenharmony_ci submit_bio(bio); 3208c2ecf20Sopenharmony_ci 3218c2ecf20Sopenharmony_ci return 0; 3228c2ecf20Sopenharmony_ci} 3238c2ecf20Sopenharmony_ci 3248c2ecf20Sopenharmony_ciint swap_readpage(struct page *page, bool synchronous) 3258c2ecf20Sopenharmony_ci{ 3268c2ecf20Sopenharmony_ci struct bio *bio; 3278c2ecf20Sopenharmony_ci int ret = 0; 3288c2ecf20Sopenharmony_ci struct swap_info_struct *sis = page_swap_info(page); 3298c2ecf20Sopenharmony_ci blk_qc_t qc; 3308c2ecf20Sopenharmony_ci struct gendisk *disk; 3318c2ecf20Sopenharmony_ci unsigned long pflags; 3328c2ecf20Sopenharmony_ci 3338c2ecf20Sopenharmony_ci VM_BUG_ON_PAGE(!PageSwapCache(page) && !synchronous, page); 3348c2ecf20Sopenharmony_ci VM_BUG_ON_PAGE(!PageLocked(page), page); 3358c2ecf20Sopenharmony_ci VM_BUG_ON_PAGE(PageUptodate(page), page); 3368c2ecf20Sopenharmony_ci 3378c2ecf20Sopenharmony_ci /* 3388c2ecf20Sopenharmony_ci * Count submission time as memory stall. When the device is congested, 3398c2ecf20Sopenharmony_ci * or the submitting cgroup IO-throttled, submission can be a 3408c2ecf20Sopenharmony_ci * significant part of overall IO time. 3418c2ecf20Sopenharmony_ci */ 3428c2ecf20Sopenharmony_ci psi_memstall_enter(&pflags); 3438c2ecf20Sopenharmony_ci 3448c2ecf20Sopenharmony_ci if (frontswap_load(page) == 0) { 3458c2ecf20Sopenharmony_ci SetPageUptodate(page); 3468c2ecf20Sopenharmony_ci unlock_page(page); 3478c2ecf20Sopenharmony_ci goto out; 3488c2ecf20Sopenharmony_ci } 3498c2ecf20Sopenharmony_ci 3508c2ecf20Sopenharmony_ci if (data_race(sis->flags & SWP_FS_OPS)) { 3518c2ecf20Sopenharmony_ci struct file *swap_file = sis->swap_file; 3528c2ecf20Sopenharmony_ci struct address_space *mapping = swap_file->f_mapping; 3538c2ecf20Sopenharmony_ci 3548c2ecf20Sopenharmony_ci ret = mapping->a_ops->readpage(swap_file, page); 3558c2ecf20Sopenharmony_ci if (!ret) 3568c2ecf20Sopenharmony_ci count_vm_event(PSWPIN); 3578c2ecf20Sopenharmony_ci goto out; 3588c2ecf20Sopenharmony_ci } 3598c2ecf20Sopenharmony_ci 3608c2ecf20Sopenharmony_ci if (sis->flags & SWP_SYNCHRONOUS_IO) { 3618c2ecf20Sopenharmony_ci ret = bdev_read_page(sis->bdev, swap_page_sector(page), page); 3628c2ecf20Sopenharmony_ci if (!ret) { 3638c2ecf20Sopenharmony_ci count_vm_event(PSWPIN); 3648c2ecf20Sopenharmony_ci goto out; 3658c2ecf20Sopenharmony_ci } 3668c2ecf20Sopenharmony_ci } 3678c2ecf20Sopenharmony_ci 3688c2ecf20Sopenharmony_ci ret = 0; 3698c2ecf20Sopenharmony_ci bio = get_swap_bio(GFP_KERNEL, page, end_swap_bio_read); 3708c2ecf20Sopenharmony_ci if (bio == NULL) { 3718c2ecf20Sopenharmony_ci unlock_page(page); 3728c2ecf20Sopenharmony_ci ret = -ENOMEM; 3738c2ecf20Sopenharmony_ci goto out; 3748c2ecf20Sopenharmony_ci } 3758c2ecf20Sopenharmony_ci disk = bio->bi_disk; 3768c2ecf20Sopenharmony_ci /* 3778c2ecf20Sopenharmony_ci * Keep this task valid during swap readpage because the oom killer may 3788c2ecf20Sopenharmony_ci * attempt to access it in the page fault retry time check. 3798c2ecf20Sopenharmony_ci */ 3808c2ecf20Sopenharmony_ci bio_set_op_attrs(bio, REQ_OP_READ, 0); 3818c2ecf20Sopenharmony_ci if (synchronous) { 3828c2ecf20Sopenharmony_ci bio->bi_opf |= REQ_HIPRI; 3838c2ecf20Sopenharmony_ci get_task_struct(current); 3848c2ecf20Sopenharmony_ci bio->bi_private = current; 3858c2ecf20Sopenharmony_ci } 3868c2ecf20Sopenharmony_ci count_vm_event(PSWPIN); 3878c2ecf20Sopenharmony_ci bio_get(bio); 3888c2ecf20Sopenharmony_ci qc = submit_bio(bio); 3898c2ecf20Sopenharmony_ci while (synchronous) { 3908c2ecf20Sopenharmony_ci set_current_state(TASK_UNINTERRUPTIBLE); 3918c2ecf20Sopenharmony_ci if (!READ_ONCE(bio->bi_private)) 3928c2ecf20Sopenharmony_ci break; 3938c2ecf20Sopenharmony_ci 3948c2ecf20Sopenharmony_ci if (!blk_poll(disk->queue, qc, true)) 3958c2ecf20Sopenharmony_ci blk_io_schedule(); 3968c2ecf20Sopenharmony_ci } 3978c2ecf20Sopenharmony_ci __set_current_state(TASK_RUNNING); 3988c2ecf20Sopenharmony_ci bio_put(bio); 3998c2ecf20Sopenharmony_ci 4008c2ecf20Sopenharmony_ciout: 4018c2ecf20Sopenharmony_ci psi_memstall_leave(&pflags); 4028c2ecf20Sopenharmony_ci return ret; 4038c2ecf20Sopenharmony_ci} 4048c2ecf20Sopenharmony_ci 4058c2ecf20Sopenharmony_ciint swap_set_page_dirty(struct page *page) 4068c2ecf20Sopenharmony_ci{ 4078c2ecf20Sopenharmony_ci struct swap_info_struct *sis = page_swap_info(page); 4088c2ecf20Sopenharmony_ci 4098c2ecf20Sopenharmony_ci if (data_race(sis->flags & SWP_FS_OPS)) { 4108c2ecf20Sopenharmony_ci struct address_space *mapping = sis->swap_file->f_mapping; 4118c2ecf20Sopenharmony_ci 4128c2ecf20Sopenharmony_ci VM_BUG_ON_PAGE(!PageSwapCache(page), page); 4138c2ecf20Sopenharmony_ci return mapping->a_ops->set_page_dirty(page); 4148c2ecf20Sopenharmony_ci } else { 4158c2ecf20Sopenharmony_ci return __set_page_dirty_no_writeback(page); 4168c2ecf20Sopenharmony_ci } 4178c2ecf20Sopenharmony_ci} 418