blob: dadb56fd0b18cc256e8776d03e8e073eef7473ca [file]
/* 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 */