2018-04-03 19:23:33 +02:00
// SPDX-License-Identifier: GPL-2.0
2007-10-15 16:18:56 -04:00
/*
* Copyright ( C ) 2007 Oracle . All rights reserved .
*/
2012-07-09 20:22:35 -06:00
# include <asm/unaligned.h>
2008-09-29 15:18:18 -04:00
2012-07-09 20:22:35 -06:00
# include "ctree.h"
2020-04-30 23:38:11 +02:00
static bool check_setget_bounds ( const struct extent_buffer * eb ,
const void * ptr , unsigned off , int size )
{
const unsigned long member_offset = ( unsigned long ) ptr + off ;
if ( member_offset > eb - > len ) {
btrfs_warn ( eb - > fs_info ,
" bad eb member start: ptr 0x%lx start %llu member offset %lu size %d " ,
( unsigned long ) ptr , eb - > start , member_offset , size ) ;
return false ;
}
if ( member_offset + size > eb - > len ) {
btrfs_warn ( eb - > fs_info ,
" bad eb member end: ptr 0x%lx start %llu member offset %lu size %d " ,
( unsigned long ) ptr , eb - > start , member_offset , size ) ;
return false ;
}
return true ;
}
2012-07-09 20:22:35 -06:00
/*
2020-05-06 20:54:13 +02:00
* Macro templates that define helpers to read / write extent buffer data of a
* given size , that are also used via ctree . h for access to item members by
* specialized helpers .
2008-09-29 15:18:18 -04:00
*
2020-05-06 20:54:13 +02:00
* Generic helpers :
* - btrfs_set_8 ( for 8 / 16 / 32 / 64 )
* - btrfs_get_8 ( for 8 / 16 / 32 / 64 )
2008-09-29 15:18:18 -04:00
*
2020-05-06 20:54:13 +02:00
* Generic helpers with a token ( cached address of the most recently accessed
* page ) :
* - btrfs_set_token_8 ( for 8 / 16 / 32 / 64 )
* - btrfs_get_token_8 ( for 8 / 16 / 32 / 64 )
2008-09-29 15:18:18 -04:00
*
2020-05-06 20:54:13 +02:00
* The set / get functions handle data spanning two pages transparently , in case
* metadata block size is larger than page . Every pointer to metadata items is
* an offset into the extent buffer page array , cast to a specific type . This
* gives us all the type checking .
2019-08-09 17:12:38 +02:00
*
2020-05-06 20:54:13 +02:00
* The extent buffer pages stored in the array pages do not form a contiguous
* phyusical range , but the API functions assume the linear offset to the range
* from 0 to metadata node size .
2008-09-29 15:18:18 -04:00
*/
2012-07-09 20:22:35 -06:00
# define DEFINE_BTRFS_SETGET_BITS(bits) \
2020-04-29 02:15:56 +02:00
u # # bits btrfs_get_token_ # # bits ( struct btrfs_map_token * token , \
const void * ptr , unsigned long off ) \
2007-10-15 16:18:56 -04:00
{ \
2020-04-29 17:45:33 +02:00
const unsigned long member_offset = ( unsigned long ) ptr + off ; \
const unsigned long idx = member_offset > > PAGE_SHIFT ; \
const unsigned long oip = offset_in_page ( member_offset ) ; \
const int size = sizeof ( u # # bits ) ; \
2020-04-30 17:57:55 +02:00
u8 lebytes [ sizeof ( u # # bits ) ] ; \
const int part = PAGE_SIZE - oip ; \
2012-07-09 20:22:35 -06:00
\
2019-08-09 17:30:23 +02:00
ASSERT ( token ) ; \
2020-04-29 19:29:04 +02:00
ASSERT ( token - > kaddr ) ; \
2020-04-30 23:38:11 +02:00
ASSERT ( check_setget_bounds ( token - > eb , ptr , off , size ) ) ; \
2020-04-29 17:45:33 +02:00
if ( token - > offset < = member_offset & & \
member_offset + size < = token - > offset + PAGE_SIZE ) { \
return get_unaligned_le # # bits ( token - > kaddr + oip ) ; \
2012-07-09 20:22:35 -06:00
} \
2020-04-30 17:57:55 +02:00
token - > kaddr = page_address ( token - > eb - > pages [ idx ] ) ; \
token - > offset = idx < < PAGE_SHIFT ; \
if ( oip + size < = PAGE_SIZE ) \
2020-04-29 17:45:33 +02:00
return get_unaligned_le # # bits ( token - > kaddr + oip ) ; \
2020-04-30 17:57:55 +02:00
\
memcpy ( lebytes , token - > kaddr + oip , part ) ; \
2020-04-29 17:45:33 +02:00
token - > kaddr = page_address ( token - > eb - > pages [ idx + 1 ] ) ; \
token - > offset = ( idx + 1 ) < < PAGE_SHIFT ; \
2020-04-30 17:57:55 +02:00
memcpy ( lebytes + part , token - > kaddr , size - part ) ; \
return get_unaligned_le # # bits ( lebytes ) ; \
2007-10-15 16:18:56 -04:00
} \
2019-08-09 17:12:38 +02:00
u # # bits btrfs_get_ # # bits ( const struct extent_buffer * eb , \
const void * ptr , unsigned long off ) \
{ \
2020-04-29 16:04:44 +02:00
const unsigned long member_offset = ( unsigned long ) ptr + off ; \
const unsigned long oip = offset_in_page ( member_offset ) ; \
2020-04-30 17:57:55 +02:00
const unsigned long idx = member_offset > > PAGE_SHIFT ; \
char * kaddr = page_address ( eb - > pages [ idx ] ) ; \
2020-04-29 16:04:44 +02:00
const int size = sizeof ( u # # bits ) ; \
2020-04-30 17:57:55 +02:00
const int part = PAGE_SIZE - oip ; \
u8 lebytes [ sizeof ( u # # bits ) ] ; \
2019-08-09 17:12:38 +02:00
\
2020-04-30 23:38:11 +02:00
ASSERT ( check_setget_bounds ( eb , ptr , off , size ) ) ; \
2020-04-30 17:57:55 +02:00
if ( oip + size < = PAGE_SIZE ) \
2020-04-29 16:04:44 +02:00
return get_unaligned_le # # bits ( kaddr + oip ) ; \
2020-04-30 17:57:55 +02:00
\
memcpy ( lebytes , kaddr + oip , part ) ; \
kaddr = page_address ( eb - > pages [ idx + 1 ] ) ; \
memcpy ( lebytes + part , kaddr , size - part ) ; \
return get_unaligned_le # # bits ( lebytes ) ; \
2019-08-09 17:12:38 +02:00
} \
2020-04-29 02:15:56 +02:00
void btrfs_set_token_ # # bits ( struct btrfs_map_token * token , \
2017-06-28 21:56:53 -06:00
const void * ptr , unsigned long off , \
2020-04-29 02:15:56 +02:00
u # # bits val ) \
2007-10-15 16:18:56 -04:00
{ \
2020-04-29 18:23:37 +02:00
const unsigned long member_offset = ( unsigned long ) ptr + off ; \
const unsigned long idx = member_offset > > PAGE_SHIFT ; \
const unsigned long oip = offset_in_page ( member_offset ) ; \
const int size = sizeof ( u # # bits ) ; \
2020-04-30 17:57:55 +02:00
u8 lebytes [ sizeof ( u # # bits ) ] ; \
const int part = PAGE_SIZE - oip ; \
2012-07-09 20:22:35 -06:00
\
2019-08-09 17:30:23 +02:00
ASSERT ( token ) ; \
2020-04-29 19:29:04 +02:00
ASSERT ( token - > kaddr ) ; \
2020-04-30 23:38:11 +02:00
ASSERT ( check_setget_bounds ( token - > eb , ptr , off , size ) ) ; \
2020-04-29 18:23:37 +02:00
if ( token - > offset < = member_offset & & \
member_offset + size < = token - > offset + PAGE_SIZE ) { \
put_unaligned_le # # bits ( val , token - > kaddr + oip ) ; \
2012-07-09 20:22:35 -06:00
return ; \
} \
2020-04-30 17:57:55 +02:00
token - > kaddr = page_address ( token - > eb - > pages [ idx ] ) ; \
token - > offset = idx < < PAGE_SHIFT ; \
2020-04-29 18:23:37 +02:00
if ( oip + size < = PAGE_SIZE ) { \
put_unaligned_le # # bits ( val , token - > kaddr + oip ) ; \
2012-07-09 20:22:35 -06:00
return ; \
} \
2020-04-30 17:57:55 +02:00
put_unaligned_le # # bits ( val , lebytes ) ; \
memcpy ( token - > kaddr + oip , lebytes , part ) ; \
2020-04-29 18:23:37 +02:00
token - > kaddr = page_address ( token - > eb - > pages [ idx + 1 ] ) ; \
token - > offset = ( idx + 1 ) < < PAGE_SHIFT ; \
2020-04-30 17:57:55 +02:00
memcpy ( token - > kaddr , lebytes + part , size - part ) ; \
2019-08-09 17:12:38 +02:00
} \
2020-04-29 03:04:10 +02:00
void btrfs_set_ # # bits ( const struct extent_buffer * eb , void * ptr , \
2019-08-09 17:12:38 +02:00
unsigned long off , u # # bits val ) \
{ \
2020-04-29 18:07:04 +02:00
const unsigned long member_offset = ( unsigned long ) ptr + off ; \
const unsigned long oip = offset_in_page ( member_offset ) ; \
2020-04-30 17:57:55 +02:00
const unsigned long idx = member_offset > > PAGE_SHIFT ; \
char * kaddr = page_address ( eb - > pages [ idx ] ) ; \
2020-04-29 18:07:04 +02:00
const int size = sizeof ( u # # bits ) ; \
2020-04-30 17:57:55 +02:00
const int part = PAGE_SIZE - oip ; \
u8 lebytes [ sizeof ( u # # bits ) ] ; \
2019-08-09 17:12:38 +02:00
\
2020-04-30 23:38:11 +02:00
ASSERT ( check_setget_bounds ( eb , ptr , off , size ) ) ; \
2020-04-29 18:07:04 +02:00
if ( oip + size < = PAGE_SIZE ) { \
put_unaligned_le # # bits ( val , kaddr + oip ) ; \
2019-08-09 17:12:38 +02:00
return ; \
} \
2020-04-30 17:57:55 +02:00
\
put_unaligned_le # # bits ( val , lebytes ) ; \
memcpy ( kaddr + oip , lebytes , part ) ; \
kaddr = page_address ( eb - > pages [ idx + 1 ] ) ; \
memcpy ( kaddr , lebytes + part , size - part ) ; \
2012-07-09 20:22:35 -06:00
}
2007-10-15 16:18:56 -04:00
2012-07-09 20:22:35 -06:00
DEFINE_BTRFS_SETGET_BITS ( 8 )
DEFINE_BTRFS_SETGET_BITS ( 16 )
DEFINE_BTRFS_SETGET_BITS ( 32 )
DEFINE_BTRFS_SETGET_BITS ( 64 )
2007-10-15 16:18:56 -04:00
2017-06-28 21:56:53 -06:00
void btrfs_node_key ( const struct extent_buffer * eb ,
2007-11-06 15:09:29 -05:00
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 ) ;
}