mirror of
git://git.yoctoproject.org/poky.git
synced 2025-07-19 21:09:03 +02:00

FIEMAP API was added to Linux kernel 2.6.28 back in 2008 SEEK_HOLE and SEEK_DATA API was added much letter. As FIEMAP is used by filemap module as a default API it's safe to remove FileMpSeek class as it's never used. [YOCTO #10618] (From OE-Core rev: 44e9406ea6e3263d2fb95e9d534a21f74f318480) Signed-off-by: Ed Bartosh <ed.bartosh@linux.intel.com> Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
352 lines
13 KiB
Python
352 lines
13 KiB
Python
# Copyright (c) 2012 Intel, Inc.
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License, version 2,
|
|
# as published by the Free Software Foundation.
|
|
#
|
|
# This program is distributed in the hope that it will be useful, but
|
|
# WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
# General Public License for more details.
|
|
|
|
"""
|
|
This module implements python implements a way to get file block. Two methods
|
|
are supported - the FIEMAP ioctl and the 'SEEK_HOLE / SEEK_DATA' features of
|
|
the file seek syscall. The former is implemented by the 'FilemapFiemap' class,
|
|
the latter is implemented by the 'FilemapSeek' class. Both classes provide the
|
|
same API. The 'filemap' function automatically selects which class can be used
|
|
and returns an instance of the class.
|
|
"""
|
|
|
|
# Disable the following pylint recommendations:
|
|
# * Too many instance attributes (R0902)
|
|
# pylint: disable=R0902
|
|
|
|
import os
|
|
import struct
|
|
import array
|
|
import fcntl
|
|
import tempfile
|
|
import logging
|
|
|
|
def get_block_size(file_obj):
|
|
"""
|
|
Returns block size for file object 'file_obj'. Errors are indicated by the
|
|
'IOError' exception.
|
|
"""
|
|
|
|
from fcntl import ioctl
|
|
import struct
|
|
|
|
# Get the block size of the host file-system for the image file by calling
|
|
# the FIGETBSZ ioctl (number 2).
|
|
binary_data = ioctl(file_obj, 2, struct.pack('I', 0))
|
|
return struct.unpack('I', binary_data)[0]
|
|
|
|
class ErrorNotSupp(Exception):
|
|
"""
|
|
An exception of this type is raised when the 'FIEMAP' or 'SEEK_HOLE' feature
|
|
is not supported either by the kernel or the file-system.
|
|
"""
|
|
pass
|
|
|
|
class Error(Exception):
|
|
"""A class for all the other exceptions raised by this module."""
|
|
pass
|
|
|
|
# Below goes the FIEMAP ioctl implementation, which is not very readable
|
|
# because it deals with the rather complex FIEMAP ioctl. To understand the
|
|
# code, you need to know the FIEMAP interface, which is documented in the
|
|
# "Documentation/filesystems/fiemap.txt" file in the Linux kernel sources.
|
|
|
|
# Format string for 'struct fiemap'
|
|
_FIEMAP_FORMAT = "=QQLLLL"
|
|
# sizeof(struct fiemap)
|
|
_FIEMAP_SIZE = struct.calcsize(_FIEMAP_FORMAT)
|
|
# Format string for 'struct fiemap_extent'
|
|
_FIEMAP_EXTENT_FORMAT = "=QQQQQLLLL"
|
|
# sizeof(struct fiemap_extent)
|
|
_FIEMAP_EXTENT_SIZE = struct.calcsize(_FIEMAP_EXTENT_FORMAT)
|
|
# The FIEMAP ioctl number
|
|
_FIEMAP_IOCTL = 0xC020660B
|
|
# This FIEMAP ioctl flag which instructs the kernel to sync the file before
|
|
# reading the block map
|
|
_FIEMAP_FLAG_SYNC = 0x00000001
|
|
# Size of the buffer for 'struct fiemap_extent' elements which will be used
|
|
# when invoking the FIEMAP ioctl. The larger is the buffer, the less times the
|
|
# FIEMAP ioctl will be invoked.
|
|
_FIEMAP_BUFFER_SIZE = 256 * 1024
|
|
|
|
class FilemapFiemap:
|
|
"""
|
|
This class provides API to the FIEMAP ioctl. Namely, it allows to iterate
|
|
over all mapped blocks and over all holes.
|
|
|
|
This class synchronizes the image file every time it invokes the FIEMAP
|
|
ioctl in order to work-around early FIEMAP implementation kernel bugs.
|
|
"""
|
|
|
|
def __init__(self, image):
|
|
"""
|
|
Initialize a class instance. The 'image' argument is full the file
|
|
object to operate on.
|
|
"""
|
|
self._log = logging.getLogger(__name__)
|
|
|
|
self._log.debug("FilemapFiemap: initializing")
|
|
|
|
self._f_image_needs_close = False
|
|
|
|
if hasattr(image, "fileno"):
|
|
self._f_image = image
|
|
self._image_path = image.name
|
|
else:
|
|
self._image_path = image
|
|
self._open_image_file()
|
|
|
|
try:
|
|
self.image_size = os.fstat(self._f_image.fileno()).st_size
|
|
except IOError as err:
|
|
raise Error("cannot get information about file '%s': %s"
|
|
% (self._f_image.name, err))
|
|
|
|
try:
|
|
self.block_size = get_block_size(self._f_image)
|
|
except IOError as err:
|
|
raise Error("cannot get block size for '%s': %s"
|
|
% (self._image_path, err))
|
|
|
|
self.blocks_cnt = self.image_size + self.block_size - 1
|
|
self.blocks_cnt //= self.block_size
|
|
|
|
try:
|
|
self._f_image.flush()
|
|
except IOError as err:
|
|
raise Error("cannot flush image file '%s': %s"
|
|
% (self._image_path, err))
|
|
|
|
try:
|
|
os.fsync(self._f_image.fileno()),
|
|
except OSError as err:
|
|
raise Error("cannot synchronize image file '%s': %s "
|
|
% (self._image_path, err.strerror))
|
|
|
|
self._log.debug("opened image \"%s\"" % self._image_path)
|
|
self._log.debug("block size %d, blocks count %d, image size %d"
|
|
% (self.block_size, self.blocks_cnt, self.image_size))
|
|
|
|
self._buf_size = _FIEMAP_BUFFER_SIZE
|
|
|
|
# Calculate how many 'struct fiemap_extent' elements fit the buffer
|
|
self._buf_size -= _FIEMAP_SIZE
|
|
self._fiemap_extent_cnt = self._buf_size // _FIEMAP_EXTENT_SIZE
|
|
assert self._fiemap_extent_cnt > 0
|
|
self._buf_size = self._fiemap_extent_cnt * _FIEMAP_EXTENT_SIZE
|
|
self._buf_size += _FIEMAP_SIZE
|
|
|
|
# Allocate a mutable buffer for the FIEMAP ioctl
|
|
self._buf = array.array('B', [0] * self._buf_size)
|
|
|
|
# Check if the FIEMAP ioctl is supported
|
|
self.block_is_mapped(0)
|
|
|
|
def __del__(self):
|
|
"""The class destructor which just closes the image file."""
|
|
if self._f_image_needs_close:
|
|
self._f_image.close()
|
|
|
|
def _open_image_file(self):
|
|
"""Open the image file."""
|
|
try:
|
|
self._f_image = open(self._image_path, 'rb')
|
|
except IOError as err:
|
|
raise Error("cannot open image file '%s': %s"
|
|
% (self._image_path, err))
|
|
|
|
self._f_image_needs_close = True
|
|
|
|
def _invoke_fiemap(self, block, count):
|
|
"""
|
|
Invoke the FIEMAP ioctl for 'count' blocks of the file starting from
|
|
block number 'block'.
|
|
|
|
The full result of the operation is stored in 'self._buf' on exit.
|
|
Returns the unpacked 'struct fiemap' data structure in form of a python
|
|
list (just like 'struct.upack()').
|
|
"""
|
|
|
|
if self.blocks_cnt != 0 and (block < 0 or block >= self.blocks_cnt):
|
|
raise Error("bad block number %d, should be within [0, %d]"
|
|
% (block, self.blocks_cnt))
|
|
|
|
# Initialize the 'struct fiemap' part of the buffer. We use the
|
|
# '_FIEMAP_FLAG_SYNC' flag in order to make sure the file is
|
|
# synchronized. The reason for this is that early FIEMAP
|
|
# implementations had many bugs related to cached dirty data, and
|
|
# synchronizing the file is a necessary work-around.
|
|
struct.pack_into(_FIEMAP_FORMAT, self._buf, 0, block * self.block_size,
|
|
count * self.block_size, _FIEMAP_FLAG_SYNC, 0,
|
|
self._fiemap_extent_cnt, 0)
|
|
|
|
try:
|
|
fcntl.ioctl(self._f_image, _FIEMAP_IOCTL, self._buf, 1)
|
|
except IOError as err:
|
|
# Note, the FIEMAP ioctl is supported by the Linux kernel starting
|
|
# from version 2.6.28 (year 2008).
|
|
if err.errno == os.errno.EOPNOTSUPP:
|
|
errstr = "FilemapFiemap: the FIEMAP ioctl is not supported " \
|
|
"by the file-system"
|
|
self._log.debug(errstr)
|
|
raise ErrorNotSupp(errstr)
|
|
if err.errno == os.errno.ENOTTY:
|
|
errstr = "FilemapFiemap: the FIEMAP ioctl is not supported " \
|
|
"by the kernel"
|
|
self._log.debug(errstr)
|
|
raise ErrorNotSupp(errstr)
|
|
raise Error("the FIEMAP ioctl failed for '%s': %s"
|
|
% (self._image_path, err))
|
|
|
|
return struct.unpack(_FIEMAP_FORMAT, self._buf[:_FIEMAP_SIZE])
|
|
|
|
def block_is_mapped(self, block):
|
|
"""Refer the '_FilemapBase' class for the documentation."""
|
|
struct_fiemap = self._invoke_fiemap(block, 1)
|
|
|
|
# The 3rd element of 'struct_fiemap' is the 'fm_mapped_extents' field.
|
|
# If it contains zero, the block is not mapped, otherwise it is
|
|
# mapped.
|
|
result = bool(struct_fiemap[3])
|
|
self._log.debug("FilemapFiemap: block_is_mapped(%d) returns %s"
|
|
% (block, result))
|
|
return result
|
|
|
|
def block_is_unmapped(self, block):
|
|
"""Refer the '_FilemapBase' class for the documentation."""
|
|
return not self.block_is_mapped(block)
|
|
|
|
def _unpack_fiemap_extent(self, index):
|
|
"""
|
|
Unpack a 'struct fiemap_extent' structure object number 'index' from
|
|
the internal 'self._buf' buffer.
|
|
"""
|
|
|
|
offset = _FIEMAP_SIZE + _FIEMAP_EXTENT_SIZE * index
|
|
return struct.unpack(_FIEMAP_EXTENT_FORMAT,
|
|
self._buf[offset : offset + _FIEMAP_EXTENT_SIZE])
|
|
|
|
def _do_get_mapped_ranges(self, start, count):
|
|
"""
|
|
Implements most the functionality for the 'get_mapped_ranges()'
|
|
generator: invokes the FIEMAP ioctl, walks through the mapped extents
|
|
and yields mapped block ranges. However, the ranges may be consecutive
|
|
(e.g., (1, 100), (100, 200)) and 'get_mapped_ranges()' simply merges
|
|
them.
|
|
"""
|
|
|
|
block = start
|
|
while block < start + count:
|
|
struct_fiemap = self._invoke_fiemap(block, count)
|
|
|
|
mapped_extents = struct_fiemap[3]
|
|
if mapped_extents == 0:
|
|
# No more mapped blocks
|
|
return
|
|
|
|
extent = 0
|
|
while extent < mapped_extents:
|
|
fiemap_extent = self._unpack_fiemap_extent(extent)
|
|
|
|
# Start of the extent
|
|
extent_start = fiemap_extent[0]
|
|
# Starting block number of the extent
|
|
extent_block = extent_start // self.block_size
|
|
# Length of the extent
|
|
extent_len = fiemap_extent[2]
|
|
# Count of blocks in the extent
|
|
extent_count = extent_len // self.block_size
|
|
|
|
# Extent length and offset have to be block-aligned
|
|
assert extent_start % self.block_size == 0
|
|
assert extent_len % self.block_size == 0
|
|
|
|
if extent_block > start + count - 1:
|
|
return
|
|
|
|
first = max(extent_block, block)
|
|
last = min(extent_block + extent_count, start + count) - 1
|
|
yield (first, last)
|
|
|
|
extent += 1
|
|
|
|
block = extent_block + extent_count
|
|
|
|
def get_mapped_ranges(self, start, count):
|
|
"""Refer the '_FilemapBase' class for the documentation."""
|
|
self._log.debug("FilemapFiemap: get_mapped_ranges(%d, %d(%d))"
|
|
% (start, count, start + count - 1))
|
|
iterator = self._do_get_mapped_ranges(start, count)
|
|
first_prev, last_prev = next(iterator)
|
|
|
|
for first, last in iterator:
|
|
if last_prev == first - 1:
|
|
last_prev = last
|
|
else:
|
|
self._log.debug("FilemapFiemap: yielding range (%d, %d)"
|
|
% (first_prev, last_prev))
|
|
yield (first_prev, last_prev)
|
|
first_prev, last_prev = first, last
|
|
|
|
self._log.debug("FilemapFiemap: yielding range (%d, %d)"
|
|
% (first_prev, last_prev))
|
|
yield (first_prev, last_prev)
|
|
|
|
def get_unmapped_ranges(self, start, count):
|
|
"""Refer the '_FilemapBase' class for the documentation."""
|
|
self._log.debug("FilemapFiemap: get_unmapped_ranges(%d, %d(%d))"
|
|
% (start, count, start + count - 1))
|
|
hole_first = start
|
|
for first, last in self._do_get_mapped_ranges(start, count):
|
|
if first > hole_first:
|
|
self._log.debug("FilemapFiemap: yielding range (%d, %d)"
|
|
% (hole_first, first - 1))
|
|
yield (hole_first, first - 1)
|
|
|
|
hole_first = last + 1
|
|
|
|
if hole_first < start + count:
|
|
self._log.debug("FilemapFiemap: yielding range (%d, %d)"
|
|
% (hole_first, start + count - 1))
|
|
yield (hole_first, start + count - 1)
|
|
|
|
|
|
def sparse_copy(src_fname, dst_fname, offset=0, skip=0):
|
|
"""Efficiently copy sparse file to or into another file."""
|
|
fmap = FilemapFiemap(src_fname)
|
|
try:
|
|
dst_file = open(dst_fname, 'r+b')
|
|
except IOError:
|
|
dst_file = open(dst_fname, 'wb')
|
|
dst_file.truncate(os.path.getsize(src_fname))
|
|
|
|
for first, last in fmap.get_mapped_ranges(0, fmap.blocks_cnt):
|
|
start = first * fmap.block_size
|
|
end = (last + 1) * fmap.block_size
|
|
|
|
if start < skip < end:
|
|
fmap._f_image.seek(skip, os.SEEK_SET)
|
|
else:
|
|
fmap._f_image.seek(start, os.SEEK_SET)
|
|
dst_file.seek(offset + start, os.SEEK_SET)
|
|
|
|
chunk_size = 1024 * 1024
|
|
to_read = end - start
|
|
read = 0
|
|
|
|
while read < to_read:
|
|
if read + chunk_size > to_read:
|
|
chunk_size = to_read - read
|
|
chunk = fmap._f_image.read(chunk_size)
|
|
dst_file.write(chunk)
|
|
read += chunk_size
|
|
dst_file.close()
|