Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 | /* SPDX-License-Identifier: GPL-2.0-only */ /* * Copyright 2023 Red Hat */ #ifndef VDO_PACKER_H #define VDO_PACKER_H #include <linux/list.h> #include "admin-state.h" #include "constants.h" #include "encodings.h" #include "statistics.h" #include "types.h" #include "wait-queue.h" enum { DEFAULT_PACKER_BINS = 16, }; /* The header of a compressed block. */ struct compressed_block_header { /* Unsigned 32-bit major and minor versions, little-endian */ struct packed_version_number version; /* List of unsigned 16-bit compressed block sizes, little-endian */ __le16 sizes[VDO_MAX_COMPRESSION_SLOTS]; } __packed; enum { VDO_COMPRESSED_BLOCK_DATA_SIZE = VDO_BLOCK_SIZE - sizeof(struct compressed_block_header), /* * A compressed block is only written if we can pack at least two fragments into it, so a * fragment which fills the entire data portion of a compressed block is too big. */ VDO_MAX_COMPRESSED_FRAGMENT_SIZE = VDO_COMPRESSED_BLOCK_DATA_SIZE - 1, }; /* * The compressed block overlay. */ struct compressed_block { struct compressed_block_header header; char data[VDO_COMPRESSED_BLOCK_DATA_SIZE]; } __packed; /* * Each packer_bin holds an incomplete batch of data_vios that only partially fill a compressed * block. The bins are kept in a ring sorted by the amount of unused space so the first bin with * enough space to hold a newly-compressed data_vio can easily be found. When the bin fills up or * is flushed, the first uncanceled data_vio in the bin is selected to be the agent for that bin. * Upon entering the packer, each data_vio already has its compressed data in the first slot of the * data_vio's compressed_block (overlaid on the data_vio's scratch_block). So the agent's fragment * is already in place. The fragments for the other uncanceled data_vios in the bin are packed into * the agent's compressed block. The agent then writes out the compressed block. If the write is * successful, the agent shares its pbn lock which each of the other data_vios in its compressed * block and sends each on its way. Finally the agent itself continues on the write path as before. * * There is one special bin which is used to hold data_vios which have been canceled and removed * from their bin by the packer. These data_vios need to wait for the canceller to rendezvous with * them and so they sit in this special bin. */ struct packer_bin { /* List links for packer.packer_bins */ struct list_head list; /* The number of items in the bin */ slot_number_t slots_used; /* The number of compressed block bytes remaining in the current batch */ size_t free_space; /* The current partial batch of data_vios, waiting for more */ struct data_vio *incoming[]; }; struct packer { /* The ID of the packer's callback thread */ thread_id_t thread_id; /* The number of bins */ block_count_t size; /* A list of all packer_bins, kept sorted by free_space */ struct list_head bins; /* * A bin to hold data_vios which were canceled out of the packer and are waiting to * rendezvous with the canceling data_vio. */ struct packer_bin *canceled_bin; /* The current flush generation */ sequence_number_t flush_generation; /* The administrative state of the packer */ struct admin_state state; /* Statistics are only updated on the packer thread, but are accessed from other threads */ struct packer_statistics statistics; }; int vdo_get_compressed_block_fragment(enum block_mapping_state mapping_state, struct compressed_block *block, u16 *fragment_offset, u16 *fragment_size); int __must_check vdo_make_packer(struct vdo *vdo, block_count_t bin_count, struct packer **packer_ptr); void vdo_free_packer(struct packer *packer); struct packer_statistics __must_check vdo_get_packer_statistics(const struct packer *packer); void vdo_attempt_packing(struct data_vio *data_vio); void vdo_flush_packer(struct packer *packer); void vdo_remove_lock_holder_from_packer(struct vdo_completion *completion); void vdo_increment_packer_flush_generation(struct packer *packer); void vdo_drain_packer(struct packer *packer, struct vdo_completion *completion); void vdo_resume_packer(struct packer *packer, struct vdo_completion *parent); void vdo_dump_packer(const struct packer *packer); #endif /* VDO_PACKER_H */ |