You've already forked linux-apfs-rw
mirror of
https://github.com/linux-apfs/linux-apfs-rw.git
synced 2026-05-01 15:01:34 -07:00
a3c23dffc3
The build is broken for the 6.16 release candidate: https://github.com/linux-apfs/linux-apfs-rw/issues/99 The problem is that the page struct no longer has a member called "index". Use folios for this instead. Signed-off-by: Ernesto A. Fernández <ernesto@corellium.com>
225 lines
5.8 KiB
C
225 lines
5.8 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (C) 2018 Ernesto A. Fernández <ernesto.mnd.fernandez@gmail.com>
|
|
*/
|
|
|
|
#include "apfs.h"
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 8, 0)
|
|
#include <linux/splice.h>
|
|
#endif
|
|
#if LINUX_VERSION_CODE < KERNEL_VERSION(4, 17, 0)
|
|
typedef int vm_fault_t;
|
|
#endif
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 15, 0)
|
|
static void wait_for_stable_page(struct page *page)
|
|
{
|
|
return folio_wait_stable(page_folio(page));
|
|
}
|
|
#endif
|
|
|
|
#if LINUX_VERSION_CODE < KERNEL_VERSION(4, 11, 0)
|
|
static vm_fault_t apfs_page_mkwrite(struct vm_area_struct *vma, struct vm_fault *vmf)
|
|
{
|
|
#else
|
|
static vm_fault_t apfs_page_mkwrite(struct vm_fault *vmf)
|
|
{
|
|
struct vm_area_struct *vma = vmf->vma;
|
|
#endif
|
|
struct page *page = vmf->page;
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 7, 0)
|
|
struct folio *folio;
|
|
#endif
|
|
struct inode *inode = file_inode(vma->vm_file);
|
|
struct super_block *sb = inode->i_sb;
|
|
struct buffer_head *bh, *head;
|
|
vm_fault_t ret = VM_FAULT_LOCKED;
|
|
unsigned int blocksize, block_start, len;
|
|
u64 size;
|
|
int err = 0;
|
|
|
|
sb_start_pagefault(inode->i_sb);
|
|
file_update_time(vma->vm_file);
|
|
|
|
err = apfs_transaction_start(sb, APFS_TRANS_REG);
|
|
if (err)
|
|
goto out;
|
|
apfs_inode_join_transaction(sb, inode);
|
|
|
|
err = apfs_inode_create_exclusive_dstream(inode);
|
|
if (err) {
|
|
apfs_err(sb, "dstream creation failed for ino 0x%llx", apfs_ino(inode));
|
|
goto out_abort;
|
|
}
|
|
|
|
lock_page(page);
|
|
wait_for_stable_page(page);
|
|
if (page->mapping != inode->i_mapping) {
|
|
ret = VM_FAULT_NOPAGE;
|
|
goto out_unlock;
|
|
}
|
|
|
|
if (!page_has_buffers(page)) {
|
|
#if LINUX_VERSION_CODE < KERNEL_VERSION(6, 7, 0)
|
|
create_empty_buffers(page, sb->s_blocksize, 0);
|
|
#else
|
|
folio = page_folio(page);
|
|
bh = folio_buffers(folio);
|
|
if (!bh)
|
|
bh = create_empty_buffers(folio, sb->s_blocksize, 0);
|
|
#endif
|
|
}
|
|
|
|
size = i_size_read(inode);
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 16, 0)
|
|
if (page_folio(page)->index == size >> PAGE_SHIFT)
|
|
#else
|
|
if (page->index == size >> PAGE_SHIFT)
|
|
#endif
|
|
len = size & ~PAGE_MASK;
|
|
else
|
|
len = PAGE_SIZE;
|
|
|
|
/* The blocks were read on the fault, mark them as unmapped for CoW */
|
|
head = page_buffers(page);
|
|
blocksize = head->b_size;
|
|
for (bh = head, block_start = 0; bh != head || !block_start;
|
|
block_start += blocksize, bh = bh->b_this_page) {
|
|
if (len > block_start) {
|
|
/* If it's not a hole, the fault read it already */
|
|
ASSERT(!buffer_mapped(bh) || buffer_uptodate(bh));
|
|
if (buffer_trans(bh))
|
|
continue;
|
|
clear_buffer_mapped(bh);
|
|
}
|
|
}
|
|
unlock_page(page); /* XXX: race? */
|
|
|
|
err = block_page_mkwrite(vma, vmf, apfs_get_new_block);
|
|
if (err) {
|
|
apfs_err(sb, "mkwrite failed for ino 0x%llx", apfs_ino(inode));
|
|
goto out_abort;
|
|
}
|
|
set_page_dirty(page);
|
|
|
|
/* An immediate commit would leave the page unlocked */
|
|
APFS_SB(sb)->s_nxi->nx_transaction.t_state |= APFS_NX_TRANS_DEFER_COMMIT;
|
|
|
|
err = apfs_transaction_commit(sb);
|
|
if (err)
|
|
goto out_unlock;
|
|
goto out;
|
|
|
|
out_unlock:
|
|
unlock_page(page);
|
|
out_abort:
|
|
apfs_transaction_abort(sb);
|
|
out:
|
|
if (err)
|
|
#if LINUX_VERSION_CODE < KERNEL_VERSION(6, 6, 0) && !RHEL_VERSION_GE(9, 5)
|
|
ret = block_page_mkwrite_return(err);
|
|
#else
|
|
ret = vmf_fs_error(err);
|
|
#endif
|
|
sb_end_pagefault(inode->i_sb);
|
|
return ret;
|
|
}
|
|
|
|
static const struct vm_operations_struct apfs_file_vm_ops = {
|
|
.fault = filemap_fault,
|
|
.map_pages = filemap_map_pages,
|
|
.page_mkwrite = apfs_page_mkwrite,
|
|
};
|
|
|
|
int apfs_file_mmap(struct file *file, struct vm_area_struct *vma)
|
|
{
|
|
struct address_space *mapping = file->f_mapping;
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 19, 0) || RHEL_VERSION_GE(9, 3)
|
|
if (!mapping->a_ops->read_folio)
|
|
#else
|
|
if (!mapping->a_ops->readpage)
|
|
#endif
|
|
return -ENOEXEC;
|
|
file_accessed(file);
|
|
vma->vm_ops = &apfs_file_vm_ops;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Just flush the whole transaction for now (TODO), since that's technically
|
|
* correct and easy to implement.
|
|
*/
|
|
int apfs_fsync(struct file *file, loff_t start, loff_t end, int datasync)
|
|
{
|
|
struct inode *inode = file->f_mapping->host;
|
|
struct super_block *sb = inode->i_sb;
|
|
|
|
return apfs_sync_fs(sb, true /* wait */);
|
|
}
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 8, 0)
|
|
static ssize_t apfs_copy_file_range(struct file *src_file, loff_t src_off,
|
|
struct file *dst_file, loff_t dst_off,
|
|
size_t len, unsigned int flags)
|
|
{
|
|
return (splice_copy_file_range(src_file, src_off,
|
|
dst_file, dst_off, len));
|
|
}
|
|
#endif
|
|
|
|
const struct file_operations apfs_file_operations = {
|
|
.llseek = generic_file_llseek,
|
|
.read_iter = generic_file_read_iter,
|
|
.write_iter = generic_file_write_iter,
|
|
.mmap = apfs_file_mmap,
|
|
.open = generic_file_open,
|
|
.fsync = apfs_fsync,
|
|
.unlocked_ioctl = apfs_file_ioctl,
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 8, 0)
|
|
.copy_file_range = apfs_copy_file_range,
|
|
#elif LINUX_VERSION_CODE >= KERNEL_VERSION(5, 3, 0)
|
|
.copy_file_range = generic_copy_file_range,
|
|
#endif
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 20, 0)
|
|
.remap_file_range = apfs_remap_file_range,
|
|
#else
|
|
.clone_file_range = apfs_clone_file_range,
|
|
#endif
|
|
|
|
#if LINUX_VERSION_CODE < KERNEL_VERSION(6, 5, 0)
|
|
.splice_read = generic_file_splice_read,
|
|
#else
|
|
.splice_read = filemap_splice_read,
|
|
#endif
|
|
.splice_write = iter_file_splice_write,
|
|
};
|
|
|
|
#if LINUX_VERSION_CODE == KERNEL_VERSION(5, 3, 0)
|
|
/*
|
|
* This is needed mainly to test clones with xfstests, so we only support the
|
|
* kernel version I use during testing. TODO: support all kernel versions.
|
|
*/
|
|
int apfs_fiemap(struct inode *inode, struct fiemap_extent_info *fieinfo, u64 start, u64 len)
|
|
{
|
|
return generic_block_fiemap(inode, fieinfo, start, len, apfs_get_block);
|
|
}
|
|
#endif
|
|
|
|
const struct inode_operations apfs_file_inode_operations = {
|
|
.getattr = apfs_getattr,
|
|
.listxattr = apfs_listxattr,
|
|
.setattr = apfs_setattr,
|
|
.update_time = apfs_update_time,
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 13, 0)
|
|
.fileattr_get = apfs_fileattr_get,
|
|
.fileattr_set = apfs_fileattr_set,
|
|
#endif
|
|
#if LINUX_VERSION_CODE == KERNEL_VERSION(5, 3, 0)
|
|
.fiemap = apfs_fiemap,
|
|
#endif
|
|
};
|