| // Copyright 2017 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // This file contains the global Blobfs structure used for constructing a Blobfs filesystem in |
| // memory. |
| |
| #ifndef ZIRCON_SYSTEM_ULIB_BLOBFS_BLOBFS_H_ |
| #define ZIRCON_SYSTEM_ULIB_BLOBFS_BLOBFS_H_ |
| |
| #ifndef __Fuchsia__ |
| #error Fuchsia-only Header |
| #endif |
| |
| #include <fuchsia/blobfs/c/fidl.h> |
| #include <fuchsia/hardware/block/c/fidl.h> |
| #include <fuchsia/io/c/fidl.h> |
| #include <lib/fzl/resizeable-vmo-mapper.h> |
| #include <lib/zx/vmo.h> |
| |
| #include <memory> |
| |
| #include <bitmap/raw-bitmap.h> |
| #include <blobfs/common.h> |
| #include <blobfs/format.h> |
| #include <blobfs/mount.h> |
| #include <block-client/cpp/block-device.h> |
| #include <block-client/cpp/block-group-registry.h> |
| #include <block-client/cpp/client.h> |
| #include <digest/digest.h> |
| #include <fbl/algorithm.h> |
| #include <fbl/auto_lock.h> |
| #include <fbl/macros.h> |
| #include <fbl/ref_counted.h> |
| #include <fbl/ref_ptr.h> |
| #include <fbl/vector.h> |
| #include <fs/journal/journal.h> |
| #include <fs/trace.h> |
| #include <fs/transaction/block_transaction.h> |
| #include <fs/vfs.h> |
| #include <fs/vnode.h> |
| #include <storage/operation/unbuffered-operations-builder.h> |
| #include <trace/event.h> |
| |
| #include "allocator/allocator.h" |
| #include "allocator/extent-reserver.h" |
| #include "allocator/node-reserver.h" |
| #include "blob-cache.h" |
| #include "directory.h" |
| #include "iterator/allocated-extent-iterator.h" |
| #include "iterator/extent-iterator.h" |
| #include "metrics.h" |
| #include "transaction-manager.h" |
| |
| namespace blobfs { |
| |
| using block_client::BlockDevice; |
| using digest::Digest; |
| using storage::OperationType; |
| using storage::UnbufferedOperationsBuilder; |
| |
| constexpr char kOutgoingDataRoot[] = "root"; |
| |
| class Blobfs : public TransactionManager { |
| public: |
| DISALLOW_COPY_ASSIGN_AND_MOVE(Blobfs); |
| |
| // Creates a blobfs object |
| static zx_status_t Create(async_dispatcher_t* dispatcher, std::unique_ptr<BlockDevice> device, |
| MountOptions* options, std::unique_ptr<Blobfs>* out); |
| |
| static std::unique_ptr<BlockDevice> Destroy(std::unique_ptr<Blobfs> blobfs); |
| |
| virtual ~Blobfs(); |
| |
| //////////////// |
| // TransactionManager's fs::TransactionHandler interface. |
| // |
| // Allows transmitting read and write transactions directly to the underlying storage. |
| |
| uint32_t FsBlockSize() const final { return kBlobfsBlockSize; } |
| |
| uint32_t DeviceBlockSize() const final { return block_info_.block_size; } |
| |
| uint64_t BlockNumberToDevice(uint64_t block_num) const final { |
| return block_num * kBlobfsBlockSize / block_info_.block_size; |
| } |
| |
| zx_status_t RunOperation(const storage::Operation& operation, storage::BlockBuffer* buffer) final; |
| |
| groupid_t BlockGroupID() final; |
| |
| block_client::BlockDevice* GetDevice() final { return block_device_.get(); } |
| |
| zx_status_t Transaction(block_fifo_request_t* requests, size_t count) final { |
| TRACE_DURATION("blobfs", "Blobfs::Transaction", "count", count); |
| return block_device_->FifoTransaction(requests, count); |
| } |
| |
| //////////////// |
| // TransactionManager's SpaceManager interface. |
| // |
| // Allows viewing and controlling the size of the underlying volume. |
| |
| const Superblock& Info() const final { return info_; } |
| zx_status_t AttachVmo(const zx::vmo& vmo, vmoid_t* out) final; |
| zx_status_t DetachVmo(vmoid_t vmoid) final; |
| zx_status_t AddInodes(fzl::ResizeableVmoMapper* node_map) final; |
| zx_status_t AddBlocks(size_t nblocks, RawBitmap* block_map) final; |
| |
| //////////////// |
| // TransactionManager interface. |
| // |
| // Allows attaching VMOs, controlling the underlying volume, and sending transactions to the |
| // underlying storage (optionally through the journal). |
| |
| BlobfsMetrics& Metrics() final { return metrics_; } |
| size_t WritebackCapacity() const final; |
| fs::Journal* journal() final; |
| Writability writability() const { return writability_; } |
| |
| //////////////// |
| // Other methods. |
| |
| // Returns the internal blobfs dispatcher. |
| async_dispatcher_t* dispatcher() { return dispatcher_; } |
| |
| bool CheckBlocksAllocated(uint64_t start_block, uint64_t end_block, |
| uint64_t* first_unset = nullptr) const { |
| return allocator_->CheckBlocksAllocated(start_block, end_block, first_unset); |
| } |
| |
| NodeFinder* GetNodeFinder() { return allocator_.get(); } |
| |
| Allocator* GetAllocator() { return allocator_.get(); } |
| |
| Inode* GetNode(uint32_t node_index) { return allocator_->GetNode(node_index); } |
| |
| // Invokes "open" on the root directory. |
| // Acts as a special-case to bootstrap filesystem mounting. |
| zx_status_t OpenRootNode(fbl::RefPtr<fs::Vnode>* out, ServeLayout layout); |
| |
| BlobCache& Cache() { return blob_cache_; } |
| |
| zx_status_t Readdir(fs::vdircookie_t* cookie, void* dirents, size_t len, size_t* out_actual); |
| |
| BlockDevice* Device() const { return block_device_.get(); } |
| |
| // Returns an unique identifier for this instance. |
| uint64_t GetFsId() const { return fs_id_; } |
| |
| using SyncCallback = fs::Vnode::SyncCallback; |
| void Sync(SyncCallback closure); |
| |
| // Frees an inode, from both the reserved map and the inode table. If the |
| // inode was allocated in the inode table, write the deleted inode out to |
| // disk. |
| void FreeInode(uint32_t node_index, storage::UnbufferedOperationsBuilder* operations, |
| fbl::Vector<storage::BufferedOperation>* trim_data); |
| |
| // Writes node data to the inode table and updates disk. |
| void PersistNode(uint32_t node_index, storage::UnbufferedOperationsBuilder* operations); |
| |
| // Adds reserved blocks to allocated bitmap and writes the bitmap out to disk. |
| void PersistBlocks(const ReservedExtent& extent, storage::UnbufferedOperationsBuilder* ops); |
| |
| private: |
| friend class BlobfsChecker; |
| |
| Blobfs(async_dispatcher_t* dispatcher, std::unique_ptr<BlockDevice> device, |
| const Superblock* info, Writability writable); |
| |
| // Terminates all internal connections, updates the "clean bit" (if writable), |
| // flushes writeback buffers, empties caches, and returns the underlying |
| // block device. |
| std::unique_ptr<BlockDevice> Reset(); |
| |
| // Does a single pass of all blobs, creating uninitialized Vnode |
| // objects for them all. |
| // |
| // By executing this function at mount, we can quickly assert |
| // either the presence or absence of a blob on the system without |
| // further scanning. |
| zx_status_t InitializeVnodes(); |
| |
| // Reloads metadata from disk. Useful when metadata on disk |
| // may have changed due to journal playback. |
| zx_status_t ReloadSuperblock(); |
| |
| // Frees blocks from the allocated map (if allocated) and updates disk if necessary. |
| void FreeExtent(const Extent& extent, storage::UnbufferedOperationsBuilder* operations, |
| fbl::Vector<storage::BufferedOperation>* trim_data); |
| |
| // Free a single node. Doesn't attempt to parse the type / traverse nodes; |
| // this function just deletes a single node. |
| void FreeNode(uint32_t node_index, storage::UnbufferedOperationsBuilder* operations); |
| |
| // Given a contiguous number of blocks after a starting block, |
| // write out the bitmap to disk for the corresponding blocks. |
| // Should only be called by PersistBlocks and FreeExtent. |
| void WriteBitmap(uint64_t nblocks, uint64_t start_block, |
| storage::UnbufferedOperationsBuilder* operations); |
| |
| // Given a node within the node map at an index, write it to disk. |
| // Should only be called by AllocateNode and FreeNode. |
| void WriteNode(uint32_t map_index, storage::UnbufferedOperationsBuilder* operations); |
| |
| // Enqueues an update for allocated inode/block counts. |
| void WriteInfo(storage::UnbufferedOperationsBuilder* operations); |
| |
| // Adds a trim operation to |trim_data|. |
| void DeleteExtent(uint64_t start_block, uint64_t num_blocks, |
| fbl::Vector<storage::BufferedOperation>* trim_data); |
| |
| // Creates an unique identifier for this instance. This is to be called only during |
| // "construction". |
| zx_status_t CreateFsId(); |
| |
| // Verifies that the contents of a blob are valid. |
| zx_status_t VerifyBlob(uint32_t node_index); |
| |
| // Updates the flags field in superblock. |
| void UpdateFlags(storage::UnbufferedOperationsBuilder* operations, uint32_t flags, bool set); |
| |
| std::unique_ptr<fs::Journal> journal_; |
| Superblock info_; |
| |
| BlobCache blob_cache_; |
| |
| async_dispatcher_t* dispatcher_ = nullptr; |
| std::unique_ptr<BlockDevice> block_device_; |
| fuchsia_hardware_block_BlockInfo block_info_ = {}; |
| block_client::BlockGroupRegistry group_registry_; |
| Writability writability_; |
| |
| std::unique_ptr<Allocator> allocator_; |
| |
| fzl::ResizeableVmoMapper info_mapping_; |
| vmoid_t info_vmoid_ = {}; |
| |
| uint64_t fs_id_ = 0; |
| |
| BlobfsMetrics metrics_ = {}; |
| }; |
| |
| } // namespace blobfs |
| |
| #endif // ZIRCON_SYSTEM_ULIB_BLOBFS_BLOBFS_H_ |