Changes in 4.9.144 Kbuild: suppress packed-not-aligned warning for default setting only disable stringop truncation warnings for now test_hexdump: use memcpy instead of strncpy kobject: Replace strncpy with memcpy unifdef: use memcpy instead of strncpy kernfs: Replace strncpy with memcpy ip_tunnel: Fix name string concatenate in __ip_tunnel_create() drm: gma500: fix logic error scsi: bfa: convert to strlcpy/strlcat staging: rts5208: fix gcc-8 logic error warning kdb: use memmove instead of overlapping memcpy x86/power/64: Use char arrays for asm function names iser: set sector for ambiguous mr status errors uprobes: Fix handle_swbp() vs. unregister() + register() race once more MIPS: ralink: Fix mt7620 nd_sd pinmux mips: fix mips_get_syscall_arg o32 check IB/mlx5: Avoid load failure due to unknown link width drm/ast: Fix incorrect free on ioregs drm: set is_master to 0 upon drm_new_set_master() failure scsi: scsi_devinfo: cleanly zero-pad devinfo strings ALSA: trident: Suppress gcc string warning scsi: csiostor: Avoid content leaks and casts kgdboc: Fix restrict error kgdboc: Fix warning with module build binder: fix proc->files use-after-free svm: Add mutex_lock to protect apic_access_page_done on AMD systems drm/mediatek: fix OF sibling-node lookup Input: xpad - quirk all PDP Xbox One gamepads Input: matrix_keypad - check for errors from of_get_named_gpio() Input: elan_i2c - add ELAN0620 to the ACPI table Input: elan_i2c - add ACPI ID for Lenovo IdeaPad 330-15ARR Input: elan_i2c - add support for ELAN0621 touchpad btrfs: Always try all copies when reading extent buffers Btrfs: fix use-after-free when dumping free space ARC: change defconfig defaults to ARCv2 arc: [devboards] Add support of NFSv3 ACL udf: Allow mounting volumes with incorrect identification strings reset: make optional functions really optional reset: core: fix reset_control_put reset: fix optional reset_control_get stubs to return NULL reset: add exported __reset_control_get, return NULL if optional reset: make device_reset_optional() really optional reset: remove remaining WARN_ON() in <linux/reset.h> mm: cleancache: fix corruption on missed inode invalidation usb: gadget: dummy: fix nonsensical comparisons net: qed: use correct strncpy() size tipc: use destination length for copy string libceph: drop len argument of *verify_authorizer_reply() libceph: no need to drop con->mutex for ->get_authorizer() libceph: store ceph_auth_handshake pointer in ceph_connection libceph: factor out __prepare_write_connect() libceph: factor out __ceph_x_decrypt() libceph: factor out encrypt_authorizer() libceph: add authorizer challenge libceph: implement CEPHX_V2 calculation mode libceph: weaken sizeof check in ceph_x_verify_authorizer_reply() libceph: check authorizer reply/challenge length before reading bpf/verifier: Add spi variable to check_stack_write() bpf/verifier: Pass instruction index to check_mem_access() and check_xadd() bpf: Prevent memory disambiguation attack wil6210: missing length check in wmi_set_ie mm/hugetlb.c: don't call region_abort if region_chg fails hugetlbfs: fix offset overflow in hugetlbfs mmap hugetlbfs: check for pgoff value overflow btrfs: validate type when reading a chunk btrfs: Verify that every chunk has corresponding block group at mount time btrfs: Refactor check_leaf function for later expansion btrfs: Check if item pointer overlaps with the item itself btrfs: Add sanity check for EXTENT_DATA when reading out leaf btrfs: Add checker for EXTENT_CSUM btrfs: Move leaf and node validation checker to tree-checker.c btrfs: struct-funcs, constify readers btrfs: tree-checker: Enhance btrfs_check_node output btrfs: tree-checker: Fix false panic for sanity test btrfs: tree-checker: Add checker for dir item btrfs: tree-checker: use %zu format string for size_t btrfs: tree-check: reduce stack consumption in check_dir_item btrfs: tree-checker: Verify block_group_item btrfs: tree-checker: Detect invalid and empty essential trees btrfs: Check that each block group has corresponding chunk at mount time btrfs: tree-checker: Check level for leaves and nodes btrfs: tree-checker: Fix misleading group system information f2fs: fix a panic caused by NULL flush_cmd_control f2fs: fix race condition in between free nid allocator/initializer f2fs: detect wrong layout f2fs: return error during fill_super f2fs: check blkaddr more accuratly before issue a bio f2fs: sanity check on sit entry f2fs: enhance sanity_check_raw_super() to avoid potential overflow f2fs: clean up with is_valid_blkaddr() f2fs: introduce and spread verify_blkaddr f2fs: fix to do sanity check with secs_per_zone f2fs: fix to do sanity check with user_block_count f2fs: Add sanity_check_inode() function f2fs: fix to do sanity check with node footer and iblocks f2fs: fix to do sanity check with block address in main area f2fs: fix missing up_read f2fs: fix to do sanity check with block address in main area v2 f2fs: free meta pages if sanity check for ckpt is failed f2fs: fix to do sanity check with cp_pack_start_sum xfs: don't fail when converting shortform attr to long form during ATTR_REPLACE hugetlbfs: fix bug in pgoff overflow checking Linux 4.9.144 Signed-off-by: Greg Kroah-Hartman <gregkh@google.com>
144 lines
4.5 KiB
C
144 lines
4.5 KiB
C
/*
|
|
* Copyright (C) 2007 Oracle. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public
|
|
* License v2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public
|
|
* License along with this program; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 021110-1307, USA.
|
|
*/
|
|
|
|
#include <linux/highmem.h>
|
|
#include <asm/unaligned.h>
|
|
|
|
#include "ctree.h"
|
|
|
|
static inline u8 get_unaligned_le8(const void *p)
|
|
{
|
|
return *(u8 *)p;
|
|
}
|
|
|
|
static inline void put_unaligned_le8(u8 val, void *p)
|
|
{
|
|
*(u8 *)p = val;
|
|
}
|
|
|
|
/*
|
|
* this is some deeply nasty code.
|
|
*
|
|
* The end result is that anyone who #includes ctree.h gets a
|
|
* declaration for the btrfs_set_foo functions and btrfs_foo functions,
|
|
* which are wrappers of btrfs_set_token_#bits functions and
|
|
* btrfs_get_token_#bits functions, which are defined in this file.
|
|
*
|
|
* These setget functions do all the extent_buffer related mapping
|
|
* required to efficiently read and write specific fields in the extent
|
|
* buffers. Every pointer to metadata items in btrfs is really just
|
|
* an unsigned long offset into the extent buffer which has been
|
|
* cast to a specific type. This gives us all the gcc type checking.
|
|
*
|
|
* The extent buffer api is used to do the page spanning work required to
|
|
* have a metadata blocksize different from the page size.
|
|
*/
|
|
|
|
#define DEFINE_BTRFS_SETGET_BITS(bits) \
|
|
u##bits btrfs_get_token_##bits(const struct extent_buffer *eb, \
|
|
const void *ptr, unsigned long off, \
|
|
struct btrfs_map_token *token) \
|
|
{ \
|
|
unsigned long part_offset = (unsigned long)ptr; \
|
|
unsigned long offset = part_offset + off; \
|
|
void *p; \
|
|
int err; \
|
|
char *kaddr; \
|
|
unsigned long map_start; \
|
|
unsigned long map_len; \
|
|
int size = sizeof(u##bits); \
|
|
u##bits res; \
|
|
\
|
|
if (token && token->kaddr && token->offset <= offset && \
|
|
token->eb == eb && \
|
|
(token->offset + PAGE_SIZE >= offset + size)) { \
|
|
kaddr = token->kaddr; \
|
|
p = kaddr + part_offset - token->offset; \
|
|
res = get_unaligned_le##bits(p + off); \
|
|
return res; \
|
|
} \
|
|
err = map_private_extent_buffer(eb, offset, size, \
|
|
&kaddr, &map_start, &map_len); \
|
|
if (err) { \
|
|
__le##bits leres; \
|
|
\
|
|
read_extent_buffer(eb, &leres, offset, size); \
|
|
return le##bits##_to_cpu(leres); \
|
|
} \
|
|
p = kaddr + part_offset - map_start; \
|
|
res = get_unaligned_le##bits(p + off); \
|
|
if (token) { \
|
|
token->kaddr = kaddr; \
|
|
token->offset = map_start; \
|
|
token->eb = eb; \
|
|
} \
|
|
return res; \
|
|
} \
|
|
void btrfs_set_token_##bits(struct extent_buffer *eb, \
|
|
const void *ptr, unsigned long off, \
|
|
u##bits val, \
|
|
struct btrfs_map_token *token) \
|
|
{ \
|
|
unsigned long part_offset = (unsigned long)ptr; \
|
|
unsigned long offset = part_offset + off; \
|
|
void *p; \
|
|
int err; \
|
|
char *kaddr; \
|
|
unsigned long map_start; \
|
|
unsigned long map_len; \
|
|
int size = sizeof(u##bits); \
|
|
\
|
|
if (token && token->kaddr && token->offset <= offset && \
|
|
token->eb == eb && \
|
|
(token->offset + PAGE_SIZE >= offset + size)) { \
|
|
kaddr = token->kaddr; \
|
|
p = kaddr + part_offset - token->offset; \
|
|
put_unaligned_le##bits(val, p + off); \
|
|
return; \
|
|
} \
|
|
err = map_private_extent_buffer(eb, offset, size, \
|
|
&kaddr, &map_start, &map_len); \
|
|
if (err) { \
|
|
__le##bits val2; \
|
|
\
|
|
val2 = cpu_to_le##bits(val); \
|
|
write_extent_buffer(eb, &val2, offset, size); \
|
|
return; \
|
|
} \
|
|
p = kaddr + part_offset - map_start; \
|
|
put_unaligned_le##bits(val, p + off); \
|
|
if (token) { \
|
|
token->kaddr = kaddr; \
|
|
token->offset = map_start; \
|
|
token->eb = eb; \
|
|
} \
|
|
}
|
|
|
|
DEFINE_BTRFS_SETGET_BITS(8)
|
|
DEFINE_BTRFS_SETGET_BITS(16)
|
|
DEFINE_BTRFS_SETGET_BITS(32)
|
|
DEFINE_BTRFS_SETGET_BITS(64)
|
|
|
|
void btrfs_node_key(const struct extent_buffer *eb,
|
|
struct btrfs_disk_key *disk_key, int nr)
|
|
{
|
|
unsigned long ptr = btrfs_node_key_ptr_offset(nr);
|
|
read_eb_member(eb, (struct btrfs_key_ptr *)ptr,
|
|
struct btrfs_key_ptr, key, disk_key);
|
|
}
|