2005-04-16 15:20:36 -07:00
/*
2007-05-09 02:33:02 -07:00
* dm - exception - store . c
2005-04-16 15:20:36 -07:00
*
* Copyright ( C ) 2001 - 2002 Sistina Software ( UK ) Limited .
2007-05-09 02:33:02 -07:00
* Copyright ( C ) 2006 Red Hat GmbH
2005-04-16 15:20:36 -07:00
*
* This file is released under the GPL .
*/
# include "dm.h"
# include "dm-snap.h"
# include <linux/mm.h>
# include <linux/pagemap.h>
# include <linux/vmalloc.h>
# include <linux/slab.h>
2008-04-24 22:02:01 +01:00
# include <linux/dm-io.h>
# include <linux/dm-kcopyd.h>
2005-04-16 15:20:36 -07:00
2006-06-26 00:27:35 -07:00
# define DM_MSG_PREFIX "snapshots"
2006-10-03 01:15:25 -07:00
# define DM_CHUNK_SIZE_DEFAULT_SECTORS 32 /* 16KB */
2006-06-26 00:27:35 -07:00
2005-04-16 15:20:36 -07:00
/*-----------------------------------------------------------------
* Persistent snapshots , by persistent we mean that the snapshot
* will survive a reboot .
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
/*
* We need to store a record of which parts of the origin have
* been copied to the snapshot device . The snapshot code
* requires that we copy exception chunks to chunk aligned areas
* of the COW store . It makes sense therefore , to store the
* metadata in chunk size blocks .
*
* There is no backward or forward compatibility implemented ,
* snapshots with different disk versions than the kernel will
* not be usable . It is expected that " lvcreate " will blank out
* the start of a fresh COW device before calling the snapshot
* constructor .
*
* The first chunk of the COW device just contains the header .
* After this there is a chunk filled with exception metadata ,
* followed by as many exception chunks as can fit in the
* metadata areas .
*
* All on disk structures are in little - endian format . The end
* of the exceptions info is indicated by an exception with a
* new_chunk of 0 , which is invalid since it would point to the
* header chunk .
*/
/*
* Magic for persistent snapshots : " SnAp " - Feeble isn ' t it .
*/
# define SNAP_MAGIC 0x70416e53
/*
* The on - disk version of the metadata .
*/
# define SNAPSHOT_DISK_VERSION 1
struct disk_header {
uint32_t magic ;
/*
* Is this snapshot valid . There is no way of recovering
* an invalid snapshot .
*/
uint32_t valid ;
/*
* Simple , incrementing version . no backward
* compatibility .
*/
uint32_t version ;
/* In sectors */
uint32_t chunk_size ;
} ;
struct disk_exception {
uint64_t old_chunk ;
uint64_t new_chunk ;
} ;
struct commit_callback {
void ( * callback ) ( void * , int success ) ;
void * context ;
} ;
/*
* The top level structure for a persistent exception store .
*/
struct pstore {
struct dm_snapshot * snap ; /* up pointer to my snapshot */
int version ;
int valid ;
uint32_t exceptions_per_area ;
/*
* Now that we have an asynchronous kcopyd there is no
* need for large chunk sizes , so it wont hurt to have a
* whole chunks worth of metadata in memory at once .
*/
void * area ;
/*
* Used to keep track of which metadata area the data in
* ' chunk ' refers to .
*/
uint32_t current_area ;
/*
* The next free chunk for an exception .
*/
uint32_t next_free ;
/*
* The index of next free exception in the current
* metadata area .
*/
uint32_t current_committed ;
atomic_t pending_count ;
uint32_t callback_count ;
struct commit_callback * callbacks ;
2007-05-09 02:33:02 -07:00
struct dm_io_client * io_client ;
2007-07-12 17:28:00 +01:00
struct workqueue_struct * metadata_wq ;
2005-04-16 15:20:36 -07:00
} ;
2007-07-12 17:26:32 +01:00
static unsigned sectors_to_pages ( unsigned sectors )
2005-04-16 15:20:36 -07:00
{
2008-04-24 21:41:50 +01:00
return DIV_ROUND_UP ( sectors , PAGE_SIZE > > 9 ) ;
2005-04-16 15:20:36 -07:00
}
static int alloc_area ( struct pstore * ps )
{
int r = - ENOMEM ;
size_t len ;
2006-06-26 00:27:18 -07:00
len = ps - > snap - > chunk_size < < SECTOR_SHIFT ;
2005-04-16 15:20:36 -07:00
/*
* Allocate the chunk_size block of memory that will hold
* a single metadata area .
*/
ps - > area = vmalloc ( len ) ;
if ( ! ps - > area )
return r ;
return 0 ;
}
static void free_area ( struct pstore * ps )
{
vfree ( ps - > area ) ;
2006-10-03 01:15:25 -07:00
ps - > area = NULL ;
2005-04-16 15:20:36 -07:00
}
2007-07-12 17:28:00 +01:00
struct mdata_req {
2008-04-24 21:43:17 +01:00
struct dm_io_region * where ;
2007-07-12 17:28:00 +01:00
struct dm_io_request * io_req ;
struct work_struct work ;
int result ;
} ;
static void do_metadata ( struct work_struct * work )
{
struct mdata_req * req = container_of ( work , struct mdata_req , work ) ;
req - > result = dm_io ( req - > io_req , 1 , req - > where , NULL ) ;
}
2005-04-16 15:20:36 -07:00
/*
* Read or write a chunk aligned and sized block of data from a device .
*/
2007-07-12 17:28:00 +01:00
static int chunk_io ( struct pstore * ps , uint32_t chunk , int rw , int metadata )
2005-04-16 15:20:36 -07:00
{
2008-04-24 21:43:17 +01:00
struct dm_io_region where = {
2007-05-09 02:33:02 -07:00
. bdev = ps - > snap - > cow - > bdev ,
. sector = ps - > snap - > chunk_size * chunk ,
. count = ps - > snap - > chunk_size ,
} ;
struct dm_io_request io_req = {
. bi_rw = rw ,
. mem . type = DM_IO_VMA ,
. mem . ptr . vma = ps - > area ,
. client = ps - > io_client ,
. notify . fn = NULL ,
} ;
2007-07-12 17:28:00 +01:00
struct mdata_req req ;
if ( ! metadata )
return dm_io ( & io_req , 1 , & where , NULL ) ;
2007-05-09 02:33:02 -07:00
2007-07-12 17:28:00 +01:00
req . where = & where ;
req . io_req = & io_req ;
/*
* Issue the synchronous I / O from a different thread
* to avoid generic_make_request recursion .
*/
INIT_WORK ( & req . work , do_metadata ) ;
queue_work ( ps - > metadata_wq , & req . work ) ;
flush_workqueue ( ps - > metadata_wq ) ;
return req . result ;
2005-04-16 15:20:36 -07:00
}
/*
* Read or write a metadata area . Remembering to skip the first
* chunk which holds the header .
*/
static int area_io ( struct pstore * ps , uint32_t area , int rw )
{
int r ;
uint32_t chunk ;
/* convert a metadata area index to a chunk index */
chunk = 1 + ( ( ps - > exceptions_per_area + 1 ) * area ) ;
2007-07-12 17:28:00 +01:00
r = chunk_io ( ps , chunk , rw , 0 ) ;
2005-04-16 15:20:36 -07:00
if ( r )
return r ;
ps - > current_area = area ;
return 0 ;
}
static int zero_area ( struct pstore * ps , uint32_t area )
{
2006-06-26 00:27:18 -07:00
memset ( ps - > area , 0 , ps - > snap - > chunk_size < < SECTOR_SHIFT ) ;
2005-04-16 15:20:36 -07:00
return area_io ( ps , area , WRITE ) ;
}
static int read_header ( struct pstore * ps , int * new_snapshot )
{
int r ;
struct disk_header * dh ;
2006-06-26 00:27:18 -07:00
chunk_t chunk_size ;
2006-10-03 01:15:25 -07:00
int chunk_size_supplied = 1 ;
2005-04-16 15:20:36 -07:00
2006-10-03 01:15:25 -07:00
/*
* Use default chunk size ( or hardsect_size , if larger ) if none supplied
*/
if ( ! ps - > snap - > chunk_size ) {
ps - > snap - > chunk_size = max ( DM_CHUNK_SIZE_DEFAULT_SECTORS ,
bdev_hardsect_size ( ps - > snap - > cow - > bdev ) > > 9 ) ;
ps - > snap - > chunk_mask = ps - > snap - > chunk_size - 1 ;
ps - > snap - > chunk_shift = ffs ( ps - > snap - > chunk_size ) - 1 ;
chunk_size_supplied = 0 ;
}
2007-05-09 02:33:02 -07:00
ps - > io_client = dm_io_client_create ( sectors_to_pages ( ps - > snap - >
chunk_size ) ) ;
if ( IS_ERR ( ps - > io_client ) )
return PTR_ERR ( ps - > io_client ) ;
2005-04-16 15:20:36 -07:00
2006-10-03 01:15:25 -07:00
r = alloc_area ( ps ) ;
if ( r )
2007-05-09 02:33:02 -07:00
return r ;
2006-10-03 01:15:25 -07:00
2007-07-12 17:28:00 +01:00
r = chunk_io ( ps , 0 , READ , 1 ) ;
2006-10-03 01:15:25 -07:00
if ( r )
2007-05-09 02:33:02 -07:00
goto bad ;
2006-10-03 01:15:25 -07:00
2005-04-16 15:20:36 -07:00
dh = ( struct disk_header * ) ps - > area ;
if ( le32_to_cpu ( dh - > magic ) = = 0 ) {
* new_snapshot = 1 ;
2006-10-03 01:15:25 -07:00
return 0 ;
}
2005-04-16 15:20:36 -07:00
2006-10-03 01:15:25 -07:00
if ( le32_to_cpu ( dh - > magic ) ! = SNAP_MAGIC ) {
DMWARN ( " Invalid or corrupt snapshot " ) ;
2005-04-16 15:20:36 -07:00
r = - ENXIO ;
2007-05-09 02:33:02 -07:00
goto bad ;
2005-04-16 15:20:36 -07:00
}
2006-10-03 01:15:25 -07:00
* new_snapshot = 0 ;
ps - > valid = le32_to_cpu ( dh - > valid ) ;
ps - > version = le32_to_cpu ( dh - > version ) ;
chunk_size = le32_to_cpu ( dh - > chunk_size ) ;
if ( ! chunk_size_supplied | | ps - > snap - > chunk_size = = chunk_size )
return 0 ;
DMWARN ( " chunk size %llu in device metadata overrides "
" table chunk size of %llu. " ,
( unsigned long long ) chunk_size ,
( unsigned long long ) ps - > snap - > chunk_size ) ;
/* We had a bogus chunk_size. Fix stuff up. */
free_area ( ps ) ;
ps - > snap - > chunk_size = chunk_size ;
ps - > snap - > chunk_mask = chunk_size - 1 ;
ps - > snap - > chunk_shift = ffs ( chunk_size ) - 1 ;
2007-05-09 02:33:02 -07:00
r = dm_io_client_resize ( sectors_to_pages ( ps - > snap - > chunk_size ) ,
ps - > io_client ) ;
2006-10-03 01:15:25 -07:00
if ( r )
return r ;
r = alloc_area ( ps ) ;
2007-05-09 02:33:02 -07:00
return r ;
2006-10-03 01:15:25 -07:00
2007-05-09 02:33:02 -07:00
bad :
2006-10-03 01:15:25 -07:00
free_area ( ps ) ;
2005-04-16 15:20:36 -07:00
return r ;
}
static int write_header ( struct pstore * ps )
{
struct disk_header * dh ;
2006-06-26 00:27:18 -07:00
memset ( ps - > area , 0 , ps - > snap - > chunk_size < < SECTOR_SHIFT ) ;
2005-04-16 15:20:36 -07:00
dh = ( struct disk_header * ) ps - > area ;
dh - > magic = cpu_to_le32 ( SNAP_MAGIC ) ;
dh - > valid = cpu_to_le32 ( ps - > valid ) ;
dh - > version = cpu_to_le32 ( ps - > version ) ;
2006-06-26 00:27:18 -07:00
dh - > chunk_size = cpu_to_le32 ( ps - > snap - > chunk_size ) ;
2005-04-16 15:20:36 -07:00
2007-07-12 17:28:00 +01:00
return chunk_io ( ps , 0 , WRITE , 1 ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Access functions for the disk exceptions , these do the endian conversions .
*/
static struct disk_exception * get_exception ( struct pstore * ps , uint32_t index )
{
2006-10-03 01:15:26 -07:00
BUG_ON ( index > = ps - > exceptions_per_area ) ;
2005-04-16 15:20:36 -07:00
return ( ( struct disk_exception * ) ps - > area ) + index ;
}
2006-10-03 01:15:26 -07:00
static void read_exception ( struct pstore * ps ,
uint32_t index , struct disk_exception * result )
2005-04-16 15:20:36 -07:00
{
2006-10-03 01:15:26 -07:00
struct disk_exception * e = get_exception ( ps , index ) ;
2005-04-16 15:20:36 -07:00
/* copy it */
result - > old_chunk = le64_to_cpu ( e - > old_chunk ) ;
result - > new_chunk = le64_to_cpu ( e - > new_chunk ) ;
}
2006-10-03 01:15:26 -07:00
static void write_exception ( struct pstore * ps ,
uint32_t index , struct disk_exception * de )
2005-04-16 15:20:36 -07:00
{
2006-10-03 01:15:26 -07:00
struct disk_exception * e = get_exception ( ps , index ) ;
2005-04-16 15:20:36 -07:00
/* copy it */
e - > old_chunk = cpu_to_le64 ( de - > old_chunk ) ;
e - > new_chunk = cpu_to_le64 ( de - > new_chunk ) ;
}
/*
* Registers the exceptions that are present in the current area .
* ' full ' is filled in to indicate if the area has been
* filled .
*/
static int insert_exceptions ( struct pstore * ps , int * full )
{
int r ;
unsigned int i ;
struct disk_exception de ;
/* presume the area is full */
* full = 1 ;
for ( i = 0 ; i < ps - > exceptions_per_area ; i + + ) {
2006-10-03 01:15:26 -07:00
read_exception ( ps , i , & de ) ;
2005-04-16 15:20:36 -07:00
/*
* If the new_chunk is pointing at the start of
* the COW device , where the first metadata area
* is we know that we ' ve hit the end of the
* exceptions . Therefore the area is not full .
*/
if ( de . new_chunk = = 0LL ) {
ps - > current_committed = i ;
* full = 0 ;
break ;
}
/*
* Keep track of the start of the free chunks .
*/
if ( ps - > next_free < = de . new_chunk )
ps - > next_free = de . new_chunk + 1 ;
/*
* Otherwise we add the exception to the snapshot .
*/
r = dm_add_exception ( ps - > snap , de . old_chunk , de . new_chunk ) ;
if ( r )
return r ;
}
return 0 ;
}
static int read_exceptions ( struct pstore * ps )
{
uint32_t area ;
int r , full = 1 ;
/*
* Keeping reading chunks and inserting exceptions until
* we find a partially full area .
*/
for ( area = 0 ; full ; area + + ) {
r = area_io ( ps , area , READ ) ;
if ( r )
return r ;
r = insert_exceptions ( ps , & full ) ;
if ( r )
return r ;
}
return 0 ;
}
2007-07-12 17:26:32 +01:00
static struct pstore * get_info ( struct exception_store * store )
2005-04-16 15:20:36 -07:00
{
return ( struct pstore * ) store - > context ;
}
static void persistent_fraction_full ( struct exception_store * store ,
sector_t * numerator , sector_t * denominator )
{
* numerator = get_info ( store ) - > next_free * store - > snap - > chunk_size ;
* denominator = get_dev_size ( store - > snap - > cow - > bdev ) ;
}
static void persistent_destroy ( struct exception_store * store )
{
struct pstore * ps = get_info ( store ) ;
2007-07-12 17:28:00 +01:00
destroy_workqueue ( ps - > metadata_wq ) ;
2007-05-09 02:33:02 -07:00
dm_io_client_destroy ( ps - > io_client ) ;
2005-04-16 15:20:36 -07:00
vfree ( ps - > callbacks ) ;
free_area ( ps ) ;
kfree ( ps ) ;
}
static int persistent_read_metadata ( struct exception_store * store )
{
2008-02-08 02:10:11 +00:00
int r , uninitialized_var ( new_snapshot ) ;
2005-04-16 15:20:36 -07:00
struct pstore * ps = get_info ( store ) ;
/*
* Read the snapshot header .
*/
r = read_header ( ps , & new_snapshot ) ;
if ( r )
return r ;
2006-06-26 00:27:18 -07:00
/*
* Now we know correct chunk_size , complete the initialisation .
*/
ps - > exceptions_per_area = ( ps - > snap - > chunk_size < < SECTOR_SHIFT ) /
sizeof ( struct disk_exception ) ;
ps - > callbacks = dm_vcalloc ( ps - > exceptions_per_area ,
sizeof ( * ps - > callbacks ) ) ;
if ( ! ps - > callbacks )
return - ENOMEM ;
2005-04-16 15:20:36 -07:00
/*
* Do we need to setup a new snapshot ?
*/
if ( new_snapshot ) {
r = write_header ( ps ) ;
if ( r ) {
DMWARN ( " write_header failed " ) ;
return r ;
}
r = zero_area ( ps , 0 ) ;
if ( r ) {
DMWARN ( " zero_area(0) failed " ) ;
return r ;
}
} else {
/*
* Sanity checks .
*/
if ( ps - > version ! = SNAPSHOT_DISK_VERSION ) {
DMWARN ( " unable to handle snapshot disk version %d " ,
ps - > version ) ;
return - EINVAL ;
}
2007-07-12 17:28:13 +01:00
/*
* Metadata are valid , but snapshot is invalidated
*/
if ( ! ps - > valid )
return 1 ;
2005-04-16 15:20:36 -07:00
/*
* Read the metadata .
*/
r = read_exceptions ( ps ) ;
if ( r )
return r ;
}
return 0 ;
}
static int persistent_prepare ( struct exception_store * store ,
2007-07-12 17:26:32 +01:00
struct dm_snap_exception * e )
2005-04-16 15:20:36 -07:00
{
struct pstore * ps = get_info ( store ) ;
uint32_t stride ;
sector_t size = get_dev_size ( store - > snap - > cow - > bdev ) ;
/* Is there enough room ? */
if ( size < ( ( ps - > next_free + 1 ) * store - > snap - > chunk_size ) )
return - ENOSPC ;
e - > new_chunk = ps - > next_free ;
/*
* Move onto the next free pending , making sure to take
* into account the location of the metadata chunks .
*/
stride = ( ps - > exceptions_per_area + 1 ) ;
if ( ( + + ps - > next_free % stride ) = = 1 )
ps - > next_free + + ;
atomic_inc ( & ps - > pending_count ) ;
return 0 ;
}
static void persistent_commit ( struct exception_store * store ,
2007-07-12 17:26:32 +01:00
struct dm_snap_exception * e ,
2005-04-16 15:20:36 -07:00
void ( * callback ) ( void * , int success ) ,
void * callback_context )
{
int r ;
unsigned int i ;
struct pstore * ps = get_info ( store ) ;
struct disk_exception de ;
struct commit_callback * cb ;
de . old_chunk = e - > old_chunk ;
de . new_chunk = e - > new_chunk ;
write_exception ( ps , ps - > current_committed + + , & de ) ;
/*
* Add the callback to the back of the array . This code
* is the only place where the callback array is
* manipulated , and we know that it will never be called
* multiple times concurrently .
*/
cb = ps - > callbacks + ps - > callback_count + + ;
cb - > callback = callback ;
cb - > context = callback_context ;
/*
* If there are no more exceptions in flight , or we have
* filled this metadata area we commit the exceptions to
* disk .
*/
if ( atomic_dec_and_test ( & ps - > pending_count ) | |
( ps - > current_committed = = ps - > exceptions_per_area ) ) {
r = area_io ( ps , ps - > current_area , WRITE ) ;
if ( r )
ps - > valid = 0 ;
2006-10-03 01:15:27 -07:00
/*
* Have we completely filled the current area ?
*/
if ( ps - > current_committed = = ps - > exceptions_per_area ) {
ps - > current_committed = 0 ;
r = zero_area ( ps , ps - > current_area + 1 ) ;
if ( r )
ps - > valid = 0 ;
}
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < ps - > callback_count ; i + + ) {
cb = ps - > callbacks + i ;
cb - > callback ( cb - > context , r = = 0 ? 1 : 0 ) ;
}
ps - > callback_count = 0 ;
}
}
static void persistent_drop ( struct exception_store * store )
{
struct pstore * ps = get_info ( store ) ;
ps - > valid = 0 ;
if ( write_header ( ps ) )
DMWARN ( " write header failed " ) ;
}
2006-10-03 01:15:25 -07:00
int dm_create_persistent ( struct exception_store * store )
2005-04-16 15:20:36 -07:00
{
struct pstore * ps ;
/* allocate the pstore */
ps = kmalloc ( sizeof ( * ps ) , GFP_KERNEL ) ;
2006-10-03 01:15:25 -07:00
if ( ! ps )
return - ENOMEM ;
2005-04-16 15:20:36 -07:00
ps - > snap = store - > snap ;
ps - > valid = 1 ;
ps - > version = SNAPSHOT_DISK_VERSION ;
2006-10-03 01:15:25 -07:00
ps - > area = NULL ;
2005-04-16 15:20:36 -07:00
ps - > next_free = 2 ; /* skipping the header and first area */
ps - > current_committed = 0 ;
ps - > callback_count = 0 ;
atomic_set ( & ps - > pending_count , 0 ) ;
2006-06-26 00:27:18 -07:00
ps - > callbacks = NULL ;
2005-04-16 15:20:36 -07:00
2007-07-12 17:28:00 +01:00
ps - > metadata_wq = create_singlethread_workqueue ( " ksnaphd " ) ;
if ( ! ps - > metadata_wq ) {
2007-07-18 00:49:08 -07:00
kfree ( ps ) ;
2007-07-12 17:28:00 +01:00
DMERR ( " couldn't start header metadata update thread " ) ;
return - ENOMEM ;
}
2005-04-16 15:20:36 -07:00
store - > destroy = persistent_destroy ;
store - > read_metadata = persistent_read_metadata ;
store - > prepare_exception = persistent_prepare ;
store - > commit_exception = persistent_commit ;
store - > drop_snapshot = persistent_drop ;
store - > fraction_full = persistent_fraction_full ;
store - > context = ps ;
return 0 ;
}
/*-----------------------------------------------------------------
* Implementation of the store for non - persistent snapshots .
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
struct transient_c {
sector_t next_free ;
} ;
static void transient_destroy ( struct exception_store * store )
{
kfree ( store - > context ) ;
}
static int transient_read_metadata ( struct exception_store * store )
{
return 0 ;
}
2007-07-12 17:26:32 +01:00
static int transient_prepare ( struct exception_store * store ,
struct dm_snap_exception * e )
2005-04-16 15:20:36 -07:00
{
struct transient_c * tc = ( struct transient_c * ) store - > context ;
sector_t size = get_dev_size ( store - > snap - > cow - > bdev ) ;
if ( size < ( tc - > next_free + store - > snap - > chunk_size ) )
return - 1 ;
e - > new_chunk = sector_to_chunk ( store - > snap , tc - > next_free ) ;
tc - > next_free + = store - > snap - > chunk_size ;
return 0 ;
}
static void transient_commit ( struct exception_store * store ,
2007-07-12 17:26:32 +01:00
struct dm_snap_exception * e ,
void ( * callback ) ( void * , int success ) ,
void * callback_context )
2005-04-16 15:20:36 -07:00
{
/* Just succeed */
callback ( callback_context , 1 ) ;
}
static void transient_fraction_full ( struct exception_store * store ,
sector_t * numerator , sector_t * denominator )
{
* numerator = ( ( struct transient_c * ) store - > context ) - > next_free ;
* denominator = get_dev_size ( store - > snap - > cow - > bdev ) ;
}
2006-10-03 01:15:25 -07:00
int dm_create_transient ( struct exception_store * store )
2005-04-16 15:20:36 -07:00
{
struct transient_c * tc ;
store - > destroy = transient_destroy ;
store - > read_metadata = transient_read_metadata ;
store - > prepare_exception = transient_prepare ;
store - > commit_exception = transient_commit ;
2006-10-03 01:15:25 -07:00
store - > drop_snapshot = NULL ;
2005-04-16 15:20:36 -07:00
store - > fraction_full = transient_fraction_full ;
tc = kmalloc ( sizeof ( struct transient_c ) , GFP_KERNEL ) ;
if ( ! tc )
return - ENOMEM ;
tc - > next_free = 0 ;
store - > context = tc ;
return 0 ;
}