#include "dm-bio-record.h"
#include <linux/compiler.h>
#include <linux/module.h>
#include <linux/device-mapper.h>
#include <linux/dm-io.h>
#include <linux/vmalloc.h>
#include <linux/sort.h>
#include <linux/rbtree.h>
#include <linux/delay.h>
#include <linux/random.h>
#include <linux/reboot.h>
#include <crypto/hash.h>
#include <crypto/skcipher.h>
#include <linux/async_tx.h>
#include <linux/dm-bufio.h>
#include "dm-audit.h"
#define DM_MSG_PREFIX …
#define DEFAULT_INTERLEAVE_SECTORS …
#define DEFAULT_JOURNAL_SIZE_FACTOR …
#define DEFAULT_SECTORS_PER_BITMAP_BIT …
#define DEFAULT_BUFFER_SECTORS …
#define DEFAULT_JOURNAL_WATERMARK …
#define DEFAULT_SYNC_MSEC …
#define DEFAULT_MAX_JOURNAL_SECTORS …
#define MIN_LOG2_INTERLEAVE_SECTORS …
#define MAX_LOG2_INTERLEAVE_SECTORS …
#define METADATA_WORKQUEUE_MAX_ACTIVE …
#define RECALC_SECTORS …
#define RECALC_WRITE_SUPER …
#define BITMAP_BLOCK_SIZE …
#define BITMAP_FLUSH_INTERVAL …
#define DISCARD_FILLER …
#define SALT_SIZE …
#define RECHECK_POOL_SIZE …
#define SB_MAGIC …
#define SB_VERSION_1 …
#define SB_VERSION_2 …
#define SB_VERSION_3 …
#define SB_VERSION_4 …
#define SB_VERSION_5 …
#define SB_VERSION_6 …
#define SB_SECTORS …
#define MAX_SECTORS_PER_BLOCK …
struct superblock { … };
#define SB_FLAG_HAVE_JOURNAL_MAC …
#define SB_FLAG_RECALCULATING …
#define SB_FLAG_DIRTY_BITMAP …
#define SB_FLAG_FIXED_PADDING …
#define SB_FLAG_FIXED_HMAC …
#define SB_FLAG_INLINE …
#define JOURNAL_ENTRY_ROUNDUP …
commit_id_t;
#define JOURNAL_MAC_PER_SECTOR …
struct journal_entry { … };
#define journal_entry_tag(ic, je) …
#if BITS_PER_LONG == 64
#define journal_entry_set_sector(je, x) …
#else
#define journal_entry_set_sector …
#endif
#define journal_entry_get_sector(je) …
#define journal_entry_is_unused(je) …
#define journal_entry_set_unused(je) …
#define journal_entry_is_inprogress(je) …
#define journal_entry_set_inprogress(je) …
#define JOURNAL_BLOCK_SECTORS …
#define JOURNAL_SECTOR_DATA …
#define JOURNAL_MAC_SIZE …
struct journal_sector { … };
#define MAX_TAG_SIZE …
#define METADATA_PADDING_SECTORS …
#define N_COMMIT_IDS …
static unsigned char prev_commit_seq(unsigned char seq)
{ … }
static unsigned char next_commit_seq(unsigned char seq)
{ … }
struct journal_node { … };
struct alg_spec { … };
struct dm_integrity_c { … };
struct dm_integrity_range { … };
struct dm_integrity_io { … };
struct journal_completion { … };
struct journal_io { … };
struct bitmap_block_status { … };
static struct kmem_cache *journal_io_cache;
#define JOURNAL_IO_MEMPOOL …
#ifdef DEBUG_PRINT
#define DEBUG_print …
#define DEBUG_bytes …
#else
#define DEBUG_print(x, ...) …
#define DEBUG_bytes(bytes, len, msg, ...) …
#endif
static void dm_integrity_map_continue(struct dm_integrity_io *dio, bool from_map);
static int dm_integrity_map_inline(struct dm_integrity_io *dio, bool from_map);
static void integrity_bio_wait(struct work_struct *w);
static void dm_integrity_dtr(struct dm_target *ti);
static void dm_integrity_io_error(struct dm_integrity_c *ic, const char *msg, int err)
{ … }
static int dm_integrity_failed(struct dm_integrity_c *ic)
{ … }
static bool dm_integrity_disable_recalculate(struct dm_integrity_c *ic)
{ … }
static commit_id_t dm_integrity_commit_id(struct dm_integrity_c *ic, unsigned int i,
unsigned int j, unsigned char seq)
{ … }
static void get_area_and_offset(struct dm_integrity_c *ic, sector_t data_sector,
sector_t *area, sector_t *offset)
{ … }
#define sector_to_block(ic, n) …
static __u64 get_metadata_sector_and_offset(struct dm_integrity_c *ic, sector_t area,
sector_t offset, unsigned int *metadata_offset)
{ … }
static sector_t get_data_sector(struct dm_integrity_c *ic, sector_t area, sector_t offset)
{ … }
static void wraparound_section(struct dm_integrity_c *ic, unsigned int *sec_ptr)
{ … }
static void sb_set_version(struct dm_integrity_c *ic)
{ … }
static int sb_mac(struct dm_integrity_c *ic, bool wr)
{ … }
static int sync_rw_sb(struct dm_integrity_c *ic, blk_opf_t opf)
{ … }
#define BITMAP_OP_TEST_ALL_SET …
#define BITMAP_OP_TEST_ALL_CLEAR …
#define BITMAP_OP_SET …
#define BITMAP_OP_CLEAR …
static bool block_bitmap_op(struct dm_integrity_c *ic, struct page_list *bitmap,
sector_t sector, sector_t n_sectors, int mode)
{ … }
static void block_bitmap_copy(struct dm_integrity_c *ic, struct page_list *dst, struct page_list *src)
{ … }
static struct bitmap_block_status *sector_to_bitmap_block(struct dm_integrity_c *ic, sector_t sector)
{ … }
static void access_journal_check(struct dm_integrity_c *ic, unsigned int section, unsigned int offset,
bool e, const char *function)
{ … }
static void page_list_location(struct dm_integrity_c *ic, unsigned int section, unsigned int offset,
unsigned int *pl_index, unsigned int *pl_offset)
{ … }
static struct journal_sector *access_page_list(struct dm_integrity_c *ic, struct page_list *pl,
unsigned int section, unsigned int offset, unsigned int *n_sectors)
{ … }
static struct journal_sector *access_journal(struct dm_integrity_c *ic, unsigned int section, unsigned int offset)
{ … }
static struct journal_entry *access_journal_entry(struct dm_integrity_c *ic, unsigned int section, unsigned int n)
{ … }
static struct journal_sector *access_journal_data(struct dm_integrity_c *ic, unsigned int section, unsigned int n)
{ … }
static void section_mac(struct dm_integrity_c *ic, unsigned int section, __u8 result[JOURNAL_MAC_SIZE])
{ … }
static void rw_section_mac(struct dm_integrity_c *ic, unsigned int section, bool wr)
{ … }
static void complete_journal_op(void *context)
{ … }
static void xor_journal(struct dm_integrity_c *ic, bool encrypt, unsigned int section,
unsigned int n_sections, struct journal_completion *comp)
{ … }
static void complete_journal_encrypt(void *data, int err)
{ … }
static bool do_crypt(bool encrypt, struct skcipher_request *req, struct journal_completion *comp)
{ … }
static void crypt_journal(struct dm_integrity_c *ic, bool encrypt, unsigned int section,
unsigned int n_sections, struct journal_completion *comp)
{ … }
static void encrypt_journal(struct dm_integrity_c *ic, bool encrypt, unsigned int section,
unsigned int n_sections, struct journal_completion *comp)
{ … }
static void complete_journal_io(unsigned long error, void *context)
{ … }
static void rw_journal_sectors(struct dm_integrity_c *ic, blk_opf_t opf,
unsigned int sector, unsigned int n_sectors,
struct journal_completion *comp)
{ … }
static void rw_journal(struct dm_integrity_c *ic, blk_opf_t opf,
unsigned int section, unsigned int n_sections,
struct journal_completion *comp)
{ … }
static void write_journal(struct dm_integrity_c *ic, unsigned int commit_start, unsigned int commit_sections)
{ … }
static void copy_from_journal(struct dm_integrity_c *ic, unsigned int section, unsigned int offset,
unsigned int n_sectors, sector_t target, io_notify_fn fn, void *data)
{ … }
static bool ranges_overlap(struct dm_integrity_range *range1, struct dm_integrity_range *range2)
{ … }
static bool add_new_range(struct dm_integrity_c *ic, struct dm_integrity_range *new_range, bool check_waiting)
{ … }
static void remove_range_unlocked(struct dm_integrity_c *ic, struct dm_integrity_range *range)
{ … }
static void remove_range(struct dm_integrity_c *ic, struct dm_integrity_range *range)
{ … }
static void wait_and_add_new_range(struct dm_integrity_c *ic, struct dm_integrity_range *new_range)
{ … }
static void add_new_range_and_wait(struct dm_integrity_c *ic, struct dm_integrity_range *new_range)
{ … }
static void init_journal_node(struct journal_node *node)
{ … }
static void add_journal_node(struct dm_integrity_c *ic, struct journal_node *node, sector_t sector)
{ … }
static void remove_journal_node(struct dm_integrity_c *ic, struct journal_node *node)
{ … }
#define NOT_FOUND …
static unsigned int find_journal_node(struct dm_integrity_c *ic, sector_t sector, sector_t *next_sector)
{ … }
static bool test_journal_node(struct dm_integrity_c *ic, unsigned int pos, sector_t sector)
{ … }
static bool find_newer_committed_node(struct dm_integrity_c *ic, struct journal_node *node)
{ … }
#define TAG_READ …
#define TAG_WRITE …
#define TAG_CMP …
static int dm_integrity_rw_tag(struct dm_integrity_c *ic, unsigned char *tag, sector_t *metadata_block,
unsigned int *metadata_offset, unsigned int total_size, int op)
{ … }
struct flush_request { … };
static void flush_notify(unsigned long error, void *fr_)
{ … }
static void dm_integrity_flush_buffers(struct dm_integrity_c *ic, bool flush_data)
{ … }
static void sleep_on_endio_wait(struct dm_integrity_c *ic)
{ … }
static void autocommit_fn(struct timer_list *t)
{ … }
static void schedule_autocommit(struct dm_integrity_c *ic)
{ … }
static void submit_flush_bio(struct dm_integrity_c *ic, struct dm_integrity_io *dio)
{ … }
static void do_endio(struct dm_integrity_c *ic, struct bio *bio)
{ … }
static void do_endio_flush(struct dm_integrity_c *ic, struct dm_integrity_io *dio)
{ … }
static void dec_in_flight(struct dm_integrity_io *dio)
{ … }
static void integrity_end_io(struct bio *bio)
{ … }
static void integrity_sector_checksum(struct dm_integrity_c *ic, sector_t sector,
const char *data, char *result)
{ … }
static noinline void integrity_recheck(struct dm_integrity_io *dio, char *checksum)
{ … }
static void integrity_metadata(struct work_struct *w)
{ … }
static inline bool dm_integrity_check_limits(struct dm_integrity_c *ic, sector_t logical_sector, struct bio *bio)
{ … }
static int dm_integrity_map(struct dm_target *ti, struct bio *bio)
{ … }
static bool __journal_read_write(struct dm_integrity_io *dio, struct bio *bio,
unsigned int journal_section, unsigned int journal_entry)
{ … }
static void dm_integrity_map_continue(struct dm_integrity_io *dio, bool from_map)
{ … }
static int dm_integrity_map_inline(struct dm_integrity_io *dio, bool from_map)
{ … }
static inline void dm_integrity_free_payload(struct dm_integrity_io *dio)
{ … }
static void dm_integrity_inline_recheck(struct work_struct *w)
{ … }
static int dm_integrity_end_io(struct dm_target *ti, struct bio *bio, blk_status_t *status)
{ … }
static void integrity_bio_wait(struct work_struct *w)
{ … }
static void pad_uncommitted(struct dm_integrity_c *ic)
{ … }
static void integrity_commit(struct work_struct *w)
{ … }
static void complete_copy_from_journal(unsigned long error, void *context)
{ … }
static void restore_last_bytes(struct dm_integrity_c *ic, struct journal_sector *js,
struct journal_entry *je)
{ … }
static void do_journal_write(struct dm_integrity_c *ic, unsigned int write_start,
unsigned int write_sections, bool from_replay)
{ … }
static void integrity_writer(struct work_struct *w)
{ … }
static void recalc_write_super(struct dm_integrity_c *ic)
{ … }
static void integrity_recalc(struct work_struct *w)
{ … }
static void integrity_recalc_inline(struct work_struct *w)
{ … }
static void bitmap_block_work(struct work_struct *w)
{ … }
static void bitmap_flush_work(struct work_struct *work)
{ … }
static void init_journal(struct dm_integrity_c *ic, unsigned int start_section,
unsigned int n_sections, unsigned char commit_seq)
{ … }
static int find_commit_seq(struct dm_integrity_c *ic, unsigned int i, unsigned int j, commit_id_t id)
{ … }
static void replay_journal(struct dm_integrity_c *ic)
{ … }
static void dm_integrity_enter_synchronous_mode(struct dm_integrity_c *ic)
{ … }
static int dm_integrity_reboot(struct notifier_block *n, unsigned long code, void *x)
{ … }
static void dm_integrity_postsuspend(struct dm_target *ti)
{ … }
static void dm_integrity_resume(struct dm_target *ti)
{ … }
static void dm_integrity_status(struct dm_target *ti, status_type_t type,
unsigned int status_flags, char *result, unsigned int maxlen)
{ … }
static int dm_integrity_iterate_devices(struct dm_target *ti,
iterate_devices_callout_fn fn, void *data)
{ … }
static void dm_integrity_io_hints(struct dm_target *ti, struct queue_limits *limits)
{ … }
static void calculate_journal_section_size(struct dm_integrity_c *ic)
{ … }
static int calculate_device_limits(struct dm_integrity_c *ic)
{ … }
static void get_provided_data_sectors(struct dm_integrity_c *ic)
{ … }
static int initialize_superblock(struct dm_integrity_c *ic,
unsigned int journal_sectors, unsigned int interleave_sectors)
{ … }
static void dm_integrity_free_page_list(struct page_list *pl)
{ … }
static struct page_list *dm_integrity_alloc_page_list(unsigned int n_pages)
{ … }
static void dm_integrity_free_journal_scatterlist(struct dm_integrity_c *ic, struct scatterlist **sl)
{ … }
static struct scatterlist **dm_integrity_alloc_journal_scatterlist(struct dm_integrity_c *ic,
struct page_list *pl)
{ … }
static void free_alg(struct alg_spec *a)
{ … }
static int get_alg_and_key(const char *arg, struct alg_spec *a, char **error, char *error_inval)
{ … }
static int get_mac(struct crypto_shash **hash, struct alg_spec *a, char **error,
char *error_alg, char *error_key)
{ … }
static int create_journal(struct dm_integrity_c *ic, char **error)
{ … }
static int dm_integrity_ctr(struct dm_target *ti, unsigned int argc, char **argv)
{ … }
static void dm_integrity_dtr(struct dm_target *ti)
{ … }
static struct target_type integrity_target = …;
static int __init dm_integrity_init(void)
{ … }
static void __exit dm_integrity_exit(void)
{ … }
module_init(…) …;
module_exit(dm_integrity_exit);
MODULE_AUTHOR(…) …;
MODULE_AUTHOR(…) …;
MODULE_DESCRIPTION(…) …;
MODULE_LICENSE(…) …;