mirror of
https://github.com/torvalds/linux.git
synced 2026-04-18 06:44:00 -04:00
Update the NTFS filesystem driver's in-memory and on-disk structures:
- Introduce the infrastructure and initial support for reparse
points and EA attribute.
- Refactor the core ntfs_inode and ntfs_volume structures to support
new features such as iomap.
- Remove the unnecessary types.h and endian.h headers.
- Reorganize the comments in headers for better readability, including
fixing warnings from checkpatch.pl.
Reviewed-by: Christoph Hellwig <hch@lst.de>
Acked-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Namjae Jeon <linkinjeon@kernel.org>
246 lines
12 KiB
C
246 lines
12 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
/*
|
|
* Defines for NTFS kernel journal (LogFile) handling.
|
|
*
|
|
* Copyright (c) 2000-2005 Anton Altaparmakov
|
|
*/
|
|
|
|
#ifndef _LINUX_NTFS_LOGFILE_H
|
|
#define _LINUX_NTFS_LOGFILE_H
|
|
|
|
#include "layout.h"
|
|
|
|
/*
|
|
* Journal (LogFile) organization:
|
|
*
|
|
* Two restart areas present in the first two pages (restart pages, one restart
|
|
* area in each page). When the volume is dismounted they should be identical,
|
|
* except for the update sequence array which usually has a different update
|
|
* sequence number.
|
|
*
|
|
* These are followed by log records organized in pages headed by a log record
|
|
* header going up to log file size. Not all pages contain log records when a
|
|
* volume is first formatted, but as the volume ages, all records will be used.
|
|
* When the log file fills up, the records at the beginning are purged (by
|
|
* modifying the oldest_lsn to a higher value presumably) and writing begins
|
|
* at the beginning of the file. Effectively, the log file is viewed as a
|
|
* circular entity.
|
|
*
|
|
* NOTE: Windows NT, 2000, and XP all use log file version 1.1 but they accept
|
|
* versions <= 1.x, including 0.-1. (Yes, that is a minus one in there!) We
|
|
* probably only want to support 1.1 as this seems to be the current version
|
|
* and we don't know how that differs from the older versions. The only
|
|
* exception is if the journal is clean as marked by the two restart pages
|
|
* then it doesn't matter whether we are on an earlier version. We can just
|
|
* reinitialize the logfile and start again with version 1.1.
|
|
*/
|
|
|
|
/* Some LogFile related constants. */
|
|
#define MaxLogFileSize 0x100000000ULL
|
|
#define DefaultLogPageSize 4096
|
|
#define MinLogRecordPages 48
|
|
|
|
/*
|
|
* Log file restart page header (begins the restart area).
|
|
*
|
|
* @magic: The magic is "RSTR".
|
|
* @usa_ofs: See ntfs_record struct definition in layout.h. When creating,
|
|
* set this to be immediately after this header structure (without any
|
|
* alignment).
|
|
* @usa_count: See ntfs_record struct definition in layout.h.
|
|
* @chkdsk_lsn: The last log file sequence number found by chkdsk. Only
|
|
* used when the magic is changed to "CHKD". Otherwise this is zero.
|
|
* @system_page_size: Byte size of system pages when the log file was
|
|
* created, has to be >= 512 and a power of 2. Use this to calculate
|
|
* the required size of the usa (usa_count) and add it to usa_ofs. Then
|
|
* verify that the result is less than the value of
|
|
* the restart_area_offset.
|
|
* @log_page_size: Byte size of log file pages, has to be >= 512 and
|
|
* a power of 2. The default is 4096 and is used when the system page
|
|
* size is between 4096 and 8192. Otherwise this is set to the system
|
|
* page size instead.
|
|
* @restart_area_offset: Byte offset from the start of this header to
|
|
* the RESTART_AREA. Value has to be aligned to 8-byte boundary. When
|
|
* creating, set this to be after the usa.
|
|
* @minor_ver: Log file minor version. Only check if major version is 1.
|
|
* @major_ver: Log file major version. We only support version 1.1.
|
|
*/
|
|
struct restart_page_header {
|
|
__le32 magic;
|
|
__le16 usa_ofs;
|
|
__le16 usa_count;
|
|
__le64 chkdsk_lsn;
|
|
__le32 system_page_size;
|
|
__le32 log_page_size;
|
|
__le16 restart_area_offset;
|
|
__le16 minor_ver;
|
|
__le16 major_ver;
|
|
} __packed;
|
|
|
|
/*
|
|
* Constant for the log client indices meaning that there are no client records
|
|
* in this particular client array. Also inside the client records themselves,
|
|
* this means that there are no client records preceding or following this one.
|
|
*/
|
|
#define LOGFILE_NO_CLIENT cpu_to_le16(0xffff)
|
|
#define LOGFILE_NO_CLIENT_CPU 0xffff
|
|
|
|
/*
|
|
* These are the so far known RESTART_AREA_* flags (16-bit) which contain
|
|
* information about the log file in which they are present.
|
|
* gcc: Force enum bit width to 16.
|
|
*/
|
|
enum {
|
|
RESTART_VOLUME_IS_CLEAN = cpu_to_le16(0x0002),
|
|
RESTART_SPACE_FILLER = cpu_to_le16(0xffff),
|
|
} __packed;
|
|
|
|
/*
|
|
* Log file restart area record. The offset of this record is found by adding
|
|
* the offset of the RESTART_PAGE_HEADER to the restart_area_offset value found
|
|
* in it. See notes at restart_area_offset above.
|
|
*
|
|
* @current_lsn: The current, i.e. last LSN inside the log when
|
|
* the restart area was last written. This happens often but what is
|
|
* the interval? Is it just fixed time or is it every time a check point
|
|
* is written or somethine else? On create set to 0.
|
|
* @log_clients: Number of log client records in the array of log client
|
|
* records which follows this restart area. Must be 1.
|
|
* @client_free_list: The index of the first free log client record in
|
|
* the array of log client records. LOGFILE_NO_CLIENT means that there
|
|
* are no free log client records in the array. If != LOGFILE_NO_CLIENT,
|
|
* check that log_clients > client_free_list. On Win2k and presumably
|
|
* earlier, on a clean volume this is != LOGFILE_NO_CLIENT, and it should
|
|
* be 0, i.e. the first (and only) client record is free and thus
|
|
* the logfile is closed and hence clean. A dirty volume would have left
|
|
* the logfile open and hence this would be LOGFILE_NO_CLIENT. On WinXP
|
|
* and presumably later, the logfile is always open, even on clean
|
|
* shutdown so this should always be LOGFILE_NO_CLIENT.
|
|
* @client_in_use_list: The index of the first in-use log client record in
|
|
* the array of log client records. LOGFILE_NO_CLIENT means that there
|
|
* are no in-use log client records in the array.
|
|
* If != LOGFILE_NO_CLIENT check that log_clients > client_in_use_list.
|
|
* On Win2k and presumably earlier, on a clean volume this is
|
|
* LOGFILE_NO_CLIENT, i.e. there are no client records in use and thus
|
|
* the logfile is closed and hence clean. A dirty volume would have left
|
|
* the logfile open and hence this would be != LOGFILE_NO_CLIENT, and it
|
|
* should be 0, i.e. the first (and only) client record is in use. On
|
|
* WinXP and presumably later, the logfile is always open, even on clean
|
|
* shutdown so this should always be 0.
|
|
* @flags: Flags modifying LFS behaviour. On Win2k and presumably earlier
|
|
* this is always 0. On WinXP and presumably later, if the logfile was
|
|
* shutdown cleanly, the second bit, RESTART_VOLUME_IS_CLEAN, is set.
|
|
* This bit is cleared when the volume is mounted by WinXP and set when
|
|
* the volume is dismounted, thus if the logfile is dirty, this bit is
|
|
* clear. Thus we don't need to check the Windows version to determine
|
|
* if the logfile is clean. Instead if the logfile is closed, we know
|
|
* it must be clean. If it is open and this bit is set, we also know
|
|
* it must be clean. If on the other hand the logfile is open and this
|
|
* bit is clear, we can be almost certain that the logfile is dirty.
|
|
* @seq_number_bits: How many bits to use for the sequence number. This
|
|
* is calculated as 67 - the number of bits required to store the logfile
|
|
* size in bytes and this can be used in with the specified file_size as
|
|
* a consistency check.
|
|
* @restart_area_length: Length of the restart area including the client
|
|
* array. Following checks required if version matches. Otherwise,
|
|
* skip them. restart_area_offset + restart_area_length has to be
|
|
* <= system_page_size. Also, restart_area_length has to be >=
|
|
* client_array_offset + (log_clients * sizeof(log client record)).
|
|
* @client_array_offset: Offset from the start of this record to the first
|
|
* log client record if versions are matched. When creating, set this
|
|
* to be after this restart area structure, aligned to 8-bytes boundary.
|
|
* If the versions do not match, this is ignored and the offset is
|
|
* assumed to be (sizeof(RESTART_AREA) + 7) & ~7, i.e. rounded up to
|
|
* first 8-byte boundary. Either way, client_array_offset has to be
|
|
* aligned to an 8-byte boundary. Also, restart_area_offset +
|
|
* client_array_offset has to be <= 510. Finally, client_array_offset +
|
|
* (log_clients * sizeof(log client record)) has to be <= system_page_size.
|
|
* On Win2k and presumably earlier, this is 0x30, i.e. immediately
|
|
* following this record. On WinXP and presumably later, this is 0x40,
|
|
* i.e. there are 16 extra bytes between this record and the client
|
|
* array. This probably means that the RESTART_AREA record is actually
|
|
* bigger in WinXP and later.
|
|
* @file_size: Usable byte size of the log file.
|
|
* If the restart_area_offset + the offset of the file_size are > 510
|
|
* then corruption has occurred. This is the very first check when
|
|
* starting with the restart_area as if it fails it means that some of
|
|
* the above values will be corrupted by the multi sector transfer
|
|
* protection. The file_size has to be rounded down to be a multiple
|
|
* of the log_page_size in the RESTART_PAGE_HEADER and then it has to be
|
|
* at least big enough to store the two restart pages and 48 (0x30) log
|
|
* record pages.
|
|
* @last_lsn_data_length: Length of data of last LSN, not including the log
|
|
* record header. On create set to 0.
|
|
* @log_record_header_length: Byte size of the log record header. If the
|
|
* version matches then check that the value of log_record_header_length
|
|
* is a multiple of 8, i.e. (log_record_header_length + 7) & ~7 ==
|
|
* log_record_header_length. When creating set it to
|
|
* sizeof(LOG_RECORD_HEADER), aligned to 8 bytes.
|
|
* @log_page_data_offset: Offset to the start of data in a log record page.
|
|
* Must be a multiple of 8. On create set it to immediately after
|
|
* the update sequence array of the log record page.
|
|
* @restart_log_open_count: A counter that gets incremented every time
|
|
* the logfile is restarted which happens at mount time when the logfile
|
|
* is opened. When creating set to a random value. Win2k sets it to
|
|
* the low 32 bits of the current system time in NTFS format (see time.h).
|
|
* @reserved: Reserved/alignment to 8-byte boundary.
|
|
*/
|
|
struct restart_area {
|
|
__le64 current_lsn;
|
|
__le16 log_clients;
|
|
__le16 client_free_list;
|
|
__le16 client_in_use_list;
|
|
__le16 flags;
|
|
__le32 seq_number_bits;
|
|
__le16 restart_area_length;
|
|
__le16 client_array_offset;
|
|
__le64 file_size;
|
|
__le32 last_lsn_data_length;
|
|
__le16 log_record_header_length;
|
|
__le16 log_page_data_offset;
|
|
__le32 restart_log_open_count;
|
|
__le32 reserved;
|
|
} __packed;
|
|
|
|
/*
|
|
* Log client record. The offset of this record is found by adding the offset
|
|
* of the RESTART_AREA to the client_array_offset value found in it.
|
|
*
|
|
* @oldest_lsn: Oldest LSN needed by this client. On create set to 0.
|
|
* @client_restart_lsn: LSN at which this client needs to restart
|
|
* the volume, i.e. the current position within the log file.
|
|
* At present, if clean this should = current_lsn in restart area but it
|
|
* probably also = current_lsn when dirty most of the time.
|
|
* At create set to 0.
|
|
* @prev_client: The offset to the previous log client record in the array
|
|
* of log client records. LOGFILE_NO_CLIENT means there is no previous
|
|
* client record, i.e. this is the first one. This is always
|
|
* LOGFILE_NO_CLIENT.
|
|
* @next_client: The offset to the next log client record in the array of
|
|
* log client records. LOGFILE_NO_CLIENT means there are no next client
|
|
* records, i.e. this is the last one. This is always LOGFILE_NO_CLIENT.
|
|
* @seq_number: On Win2k and presumably earlier, this is set to zero every
|
|
* time the logfile is restarted and it is incremented when the logfile
|
|
* is closed at dismount time. Thus it is 0 when dirty and 1 when clean.
|
|
* On WinXP and presumably later, this is always 0.
|
|
* @reserved[6]: Reserved/alignment.
|
|
* @client_name_length: Length of client name in bytes. Should always be 8.
|
|
* @client_name[64]: Name of the client in Unicode. Should always be "NTFS"
|
|
* with the remaining bytes set to 0.
|
|
*/
|
|
struct log_client_record {
|
|
__le64 oldest_lsn;
|
|
__le64 client_restart_lsn;
|
|
__le16 prev_client;
|
|
__le16 next_client;
|
|
__le16 seq_number;
|
|
u8 reserved[6];
|
|
__le32 client_name_length;
|
|
__le16 client_name[64];
|
|
} __packed;
|
|
|
|
bool ntfs_check_logfile(struct inode *log_vi,
|
|
struct restart_page_header **rp);
|
|
bool ntfs_empty_logfile(struct inode *log_vi);
|
|
#endif /* _LINUX_NTFS_LOGFILE_H */
|