2005-04-17 02:20:36 +04:00
/*
* misc . c
*
* PURPOSE
* Miscellaneous routines for the OSTA - UDF ( tm ) filesystem .
*
* COPYRIGHT
* This file is distributed under the terms of the GNU General Public
* License ( GPL ) . Copies of the GPL can be obtained from :
* ftp : //prep.ai.mit.edu/pub/gnu/GPL
* Each contributing author retains all rights to their own work .
*
* ( C ) 1998 Dave Boynton
* ( C ) 1998 - 2004 Ben Fennema
* ( C ) 1999 - 2000 Stelias Computing Inc
*
* HISTORY
*
* 04 / 19 / 99 blf partial support for reading / writing specific EA ' s
*/
# include "udfdecl.h"
# include <linux/fs.h>
# include <linux/string.h>
# include <linux/udf_fs.h>
# include <linux/buffer_head.h>
# include "udf_i.h"
# include "udf_sb.h"
2007-07-19 12:47:43 +04:00
struct buffer_head * udf_tgetblk ( struct super_block * sb , int block )
2005-04-17 02:20:36 +04:00
{
if ( UDF_QUERY_FLAG ( sb , UDF_FLAG_VARCONV ) )
return sb_getblk ( sb , udf_fixed_to_variable ( block ) ) ;
else
return sb_getblk ( sb , block ) ;
}
2007-07-19 12:47:43 +04:00
struct buffer_head * udf_tread ( struct super_block * sb , int block )
2005-04-17 02:20:36 +04:00
{
if ( UDF_QUERY_FLAG ( sb , UDF_FLAG_VARCONV ) )
return sb_bread ( sb , udf_fixed_to_variable ( block ) ) ;
else
return sb_bread ( sb , block ) ;
}
2007-07-19 12:47:43 +04:00
struct genericFormat * udf_add_extendedattr ( struct inode * inode , uint32_t size ,
uint32_t type , uint8_t loc )
2005-04-17 02:20:36 +04:00
{
uint8_t * ea = NULL , * ad = NULL ;
int offset ;
uint16_t crclen ;
int i ;
ea = UDF_I_DATA ( inode ) ;
2007-07-21 15:37:18 +04:00
if ( UDF_I_LENEATTR ( inode ) ) {
2005-04-17 02:20:36 +04:00
ad = UDF_I_DATA ( inode ) + UDF_I_LENEATTR ( inode ) ;
2007-07-21 15:37:18 +04:00
} else {
2005-04-17 02:20:36 +04:00
ad = ea ;
size + = sizeof ( struct extendedAttrHeaderDesc ) ;
}
offset = inode - > i_sb - > s_blocksize - udf_file_entry_alloc_offset ( inode ) -
2007-07-21 15:37:18 +04:00
UDF_I_LENALLOC ( inode ) ;
2005-04-17 02:20:36 +04:00
/* TODO - Check for FreeEASpace */
2007-07-19 12:47:43 +04:00
if ( loc & 0x01 & & offset > = size ) {
2005-04-17 02:20:36 +04:00
struct extendedAttrHeaderDesc * eahd ;
eahd = ( struct extendedAttrHeaderDesc * ) ea ;
2007-07-19 12:47:43 +04:00
if ( UDF_I_LENALLOC ( inode ) ) {
2005-04-17 02:20:36 +04:00
memmove ( & ad [ size ] , ad , UDF_I_LENALLOC ( inode ) ) ;
}
2007-07-19 12:47:43 +04:00
if ( UDF_I_LENEATTR ( inode ) ) {
2005-04-17 02:20:36 +04:00
/* check checksum/crc */
2007-07-21 15:37:18 +04:00
if ( le16_to_cpu ( eahd - > descTag . tagIdent ) ! = TAG_IDENT_EAHD | |
le32_to_cpu ( eahd - > descTag . tagLocation ) ! = UDF_I_LOCATION ( inode ) . logicalBlockNum ) {
2005-04-17 02:20:36 +04:00
return NULL ;
}
2007-07-19 12:47:43 +04:00
} else {
2005-04-17 02:20:36 +04:00
size - = sizeof ( struct extendedAttrHeaderDesc ) ;
2007-07-21 15:37:18 +04:00
UDF_I_LENEATTR ( inode ) + = sizeof ( struct extendedAttrHeaderDesc ) ;
2005-04-17 02:20:36 +04:00
eahd - > descTag . tagIdent = cpu_to_le16 ( TAG_IDENT_EAHD ) ;
if ( UDF_SB_UDFREV ( inode - > i_sb ) > = 0x0200 )
eahd - > descTag . descVersion = cpu_to_le16 ( 3 ) ;
else
eahd - > descTag . descVersion = cpu_to_le16 ( 2 ) ;
2007-07-21 15:37:18 +04:00
eahd - > descTag . tagSerialNum = cpu_to_le16 ( UDF_SB_SERIALNUM ( inode - > i_sb ) ) ;
eahd - > descTag . tagLocation = cpu_to_le32 ( UDF_I_LOCATION ( inode ) . logicalBlockNum ) ;
2005-04-17 02:20:36 +04:00
eahd - > impAttrLocation = cpu_to_le32 ( 0xFFFFFFFF ) ;
eahd - > appAttrLocation = cpu_to_le32 ( 0xFFFFFFFF ) ;
}
offset = UDF_I_LENEATTR ( inode ) ;
2007-07-19 12:47:43 +04:00
if ( type < 2048 ) {
2007-07-21 15:37:18 +04:00
if ( le32_to_cpu ( eahd - > appAttrLocation ) < UDF_I_LENEATTR ( inode ) ) {
uint32_t aal = le32_to_cpu ( eahd - > appAttrLocation ) ;
memmove ( & ea [ offset - aal + size ] ,
& ea [ aal ] , offset - aal ) ;
2005-04-17 02:20:36 +04:00
offset - = aal ;
eahd - > appAttrLocation = cpu_to_le32 ( aal + size ) ;
}
2007-07-21 15:37:18 +04:00
if ( le32_to_cpu ( eahd - > impAttrLocation ) < UDF_I_LENEATTR ( inode ) ) {
uint32_t ial = le32_to_cpu ( eahd - > impAttrLocation ) ;
memmove ( & ea [ offset - ial + size ] ,
& ea [ ial ] , offset - ial ) ;
2005-04-17 02:20:36 +04:00
offset - = ial ;
eahd - > impAttrLocation = cpu_to_le32 ( ial + size ) ;
}
2007-07-19 12:47:43 +04:00
} else if ( type < 65536 ) {
2007-07-21 15:37:18 +04:00
if ( le32_to_cpu ( eahd - > appAttrLocation ) < UDF_I_LENEATTR ( inode ) ) {
uint32_t aal = le32_to_cpu ( eahd - > appAttrLocation ) ;
memmove ( & ea [ offset - aal + size ] ,
& ea [ aal ] , offset - aal ) ;
2005-04-17 02:20:36 +04:00
offset - = aal ;
eahd - > appAttrLocation = cpu_to_le32 ( aal + size ) ;
}
}
/* rewrite CRC + checksum of eahd */
crclen = sizeof ( struct extendedAttrHeaderDesc ) - sizeof ( tag ) ;
eahd - > descTag . descCRCLength = cpu_to_le16 ( crclen ) ;
2007-07-21 15:37:18 +04:00
eahd - > descTag . descCRC = cpu_to_le16 ( udf_crc ( ( char * ) eahd +
sizeof ( tag ) , crclen , 0 ) ) ;
2005-04-17 02:20:36 +04:00
eahd - > descTag . tagChecksum = 0 ;
2007-07-19 12:47:43 +04:00
for ( i = 0 ; i < 16 ; i + + )
2005-04-17 02:20:36 +04:00
if ( i ! = 4 )
2007-07-21 15:37:18 +04:00
eahd - > descTag . tagChecksum + = ( ( uint8_t * ) & ( eahd - > descTag ) ) [ i ] ;
2005-04-17 02:20:36 +04:00
UDF_I_LENEATTR ( inode ) + = size ;
return ( struct genericFormat * ) & ea [ offset ] ;
}
2007-07-19 12:47:43 +04:00
if ( loc & 0x02 ) {
2005-04-17 02:20:36 +04:00
}
2007-07-21 15:37:18 +04:00
2005-04-17 02:20:36 +04:00
return NULL ;
}
2007-07-19 12:47:43 +04:00
struct genericFormat * udf_get_extendedattr ( struct inode * inode , uint32_t type ,
uint8_t subtype )
2005-04-17 02:20:36 +04:00
{
struct genericFormat * gaf ;
uint8_t * ea = NULL ;
uint32_t offset ;
ea = UDF_I_DATA ( inode ) ;
2007-07-19 12:47:43 +04:00
if ( UDF_I_LENEATTR ( inode ) ) {
2005-04-17 02:20:36 +04:00
struct extendedAttrHeaderDesc * eahd ;
eahd = ( struct extendedAttrHeaderDesc * ) ea ;
/* check checksum/crc */
if ( le16_to_cpu ( eahd - > descTag . tagIdent ) ! = TAG_IDENT_EAHD | |
2007-07-21 15:37:18 +04:00
le32_to_cpu ( eahd - > descTag . tagLocation ) ! = UDF_I_LOCATION ( inode ) . logicalBlockNum ) {
2005-04-17 02:20:36 +04:00
return NULL ;
}
2007-07-19 12:47:43 +04:00
2005-04-17 02:20:36 +04:00
if ( type < 2048 )
offset = sizeof ( struct extendedAttrHeaderDesc ) ;
else if ( type < 65536 )
offset = le32_to_cpu ( eahd - > impAttrLocation ) ;
else
offset = le32_to_cpu ( eahd - > appAttrLocation ) ;
2007-07-19 12:47:43 +04:00
while ( offset < UDF_I_LENEATTR ( inode ) ) {
2005-04-17 02:20:36 +04:00
gaf = ( struct genericFormat * ) & ea [ offset ] ;
2007-07-21 15:37:18 +04:00
if ( le32_to_cpu ( gaf - > attrType ) = = type & & gaf - > attrSubtype = = subtype )
2005-04-17 02:20:36 +04:00
return gaf ;
else
offset + = le32_to_cpu ( gaf - > attrLength ) ;
}
}
2007-07-21 15:37:18 +04:00
2005-04-17 02:20:36 +04:00
return NULL ;
}
/*
* udf_read_tagged
*
* PURPOSE
* Read the first block of a tagged descriptor .
*
* HISTORY
* July 1 , 1997 - Andrew E . Mileski
* Written , tested , and released .
*/
2007-07-19 12:47:43 +04:00
struct buffer_head * udf_read_tagged ( struct super_block * sb , uint32_t block ,
uint32_t location , uint16_t * ident )
2005-04-17 02:20:36 +04:00
{
tag * tag_p ;
struct buffer_head * bh = NULL ;
register uint8_t checksum ;
register int i ;
/* Read the block */
if ( block = = 0xFFFFFFFF )
return NULL ;
bh = udf_tread ( sb , block + UDF_SB_SESSION ( sb ) ) ;
2007-07-19 12:47:43 +04:00
if ( ! bh ) {
udf_debug ( " block=%d, location=%d: read failed \n " ,
block + UDF_SB_SESSION ( sb ) , location ) ;
2005-04-17 02:20:36 +04:00
return NULL ;
}
2007-07-21 15:37:18 +04:00
tag_p = ( tag * ) ( bh - > b_data ) ;
2005-04-17 02:20:36 +04:00
* ident = le16_to_cpu ( tag_p - > tagIdent ) ;
2007-07-19 12:47:43 +04:00
if ( location ! = le32_to_cpu ( tag_p - > tagLocation ) ) {
2005-04-17 02:20:36 +04:00
udf_debug ( " location mismatch block %u, tag %u != %u \n " ,
2007-07-21 15:37:18 +04:00
block + UDF_SB_SESSION ( sb ) , le32_to_cpu ( tag_p - > tagLocation ) , location ) ;
2005-04-17 02:20:36 +04:00
goto error_out ;
}
2007-07-19 12:47:43 +04:00
2005-04-17 02:20:36 +04:00
/* Verify the tag checksum */
checksum = 0U ;
for ( i = 0 ; i < 4 ; i + + )
2007-07-21 15:37:18 +04:00
checksum + = ( uint8_t ) ( bh - > b_data [ i ] ) ;
2005-04-17 02:20:36 +04:00
for ( i = 5 ; i < 16 ; i + + )
2007-07-21 15:37:18 +04:00
checksum + = ( uint8_t ) ( bh - > b_data [ i ] ) ;
2005-04-17 02:20:36 +04:00
if ( checksum ! = tag_p - > tagChecksum ) {
printk ( KERN_ERR " udf: tag checksum failed block %d \n " , block ) ;
goto error_out ;
}
/* Verify the tag version */
if ( le16_to_cpu ( tag_p - > descVersion ) ! = 0x0002U & &
2007-07-19 12:47:43 +04:00
le16_to_cpu ( tag_p - > descVersion ) ! = 0x0003U ) {
2005-04-17 02:20:36 +04:00
udf_debug ( " tag version 0x%04x != 0x0002 || 0x0003 block %d \n " ,
2007-07-19 12:47:43 +04:00
le16_to_cpu ( tag_p - > descVersion ) , block ) ;
2005-04-17 02:20:36 +04:00
goto error_out ;
}
/* Verify the descriptor CRC */
if ( le16_to_cpu ( tag_p - > descCRCLength ) + sizeof ( tag ) > sb - > s_blocksize | |
2007-07-19 12:47:43 +04:00
le16_to_cpu ( tag_p - > descCRC ) = = udf_crc ( bh - > b_data + sizeof ( tag ) ,
2007-07-21 15:37:18 +04:00
le16_to_cpu ( tag_p - > descCRCLength ) , 0 ) ) {
2005-04-17 02:20:36 +04:00
return bh ;
}
udf_debug ( " Crc failure block %d: crc = %d, crclen = %d \n " ,
2007-07-19 12:47:43 +04:00
block + UDF_SB_SESSION ( sb ) , le16_to_cpu ( tag_p - > descCRC ) ,
le16_to_cpu ( tag_p - > descCRCLength ) ) ;
2005-04-17 02:20:36 +04:00
2007-07-21 15:37:18 +04:00
error_out :
2005-04-17 02:20:36 +04:00
brelse ( bh ) ;
return NULL ;
}
2007-07-19 12:47:43 +04:00
struct buffer_head * udf_read_ptagged ( struct super_block * sb , kernel_lb_addr loc ,
uint32_t offset , uint16_t * ident )
2005-04-17 02:20:36 +04:00
{
return udf_read_tagged ( sb , udf_get_lb_pblock ( sb , loc , offset ) ,
2007-07-19 12:47:43 +04:00
loc . logicalBlockNum + offset , ident ) ;
2005-04-17 02:20:36 +04:00
}
void udf_update_tag ( char * data , int length )
{
2007-07-21 15:37:18 +04:00
tag * tptr = ( tag * ) data ;
2005-04-17 02:20:36 +04:00
int i ;
length - = sizeof ( tag ) ;
tptr - > tagChecksum = 0 ;
tptr - > descCRCLength = cpu_to_le16 ( length ) ;
tptr - > descCRC = cpu_to_le16 ( udf_crc ( data + sizeof ( tag ) , length , 0 ) ) ;
2007-07-19 12:47:43 +04:00
for ( i = 0 ; i < 16 ; i + + )
2005-04-17 02:20:36 +04:00
if ( i ! = 4 )
2007-07-21 15:37:18 +04:00
tptr - > tagChecksum + = ( uint8_t ) ( data [ i ] ) ;
2005-04-17 02:20:36 +04:00
}
void udf_new_tag ( char * data , uint16_t ident , uint16_t version , uint16_t snum ,
2007-07-19 12:47:43 +04:00
uint32_t loc , int length )
2005-04-17 02:20:36 +04:00
{
2007-07-21 15:37:18 +04:00
tag * tptr = ( tag * ) data ;
2005-04-17 02:20:36 +04:00
tptr - > tagIdent = cpu_to_le16 ( ident ) ;
tptr - > descVersion = cpu_to_le16 ( version ) ;
tptr - > tagSerialNum = cpu_to_le16 ( snum ) ;
tptr - > tagLocation = cpu_to_le32 ( loc ) ;
udf_update_tag ( data , length ) ;
}