/cacharr

Matlab class that allows to work with very large matrices through caching; thus avoid matrix allocation out-of-memory error.

Primary LanguageMatlabGNU Lesser General Public License v3.0LGPL-3.0

CachedNDArray - Matlab N-dimensional array with caching possiblity

Content

Short description

CachedNDArray - data structure that allows to deal with large N-dimensional arrays through caching method:

  • Allows to avoid Matlab out of memory error by caching large array into several files on hard disk and then reading the necessary chunks using memmapfile function.
  • The data structure is inhereted from handle abstract class which avoids parameter-by-value and supports parameter-by-reference.
  • Supports two types of movements - continious (very slow) and discrete (fast); the former might have no more than two file to represent a chunk; while the latter means the data is processed chunk-after-chunk, and each chunk is represented strictly as a single file.
  • Caching flag can be set to either manual or automatic mode. If no caching is needed to perform, the CachedNDArray is treated like a normal Matlab array.
  • Automatic or manual dimension breakage into number of chunks.

Quick start

Use the provided test script test_CachedNDArray.m in order to run an example. The provided test includes small scale example (using small array) and a large-scale example when comparing discreet and continious caching for read/write operators. Note, depending on characteristics of your machine, it can take a while for the script to end since the large-scale example works with a 4D array of about 19Gb size in total (thus breaking the array into four 4.8Gb files).

Class description, specifics and usage

Main principle behind the caching through Matlab

The necessity for this project was caused by a common Matlab error when trying to allocate a large N-dimensional array. This was especially relevant in the context of the cryo3d project - a pipeline for fast 3D protein reconstruction from cryo-EM images.

The cryo3d pipeline, when run with very fine parameters, requires large computational resources and memory which are not always available. And it caused the mentioned out-of-memory error.

The main idea of the caching data type is to keep that large N-dimensional array on the disk instead of Matlab memory, and only keep its smaller part that is referred to when needed. It can also be explained by using the figure:

Alt text

When we try to allocate a variable which takes more space than what we have in Matlab's available memory, the Matlab throws out-of-memory error. With the caching variable, we save the data to the disk by breaking the data into smaller chunks and saving to different files as shown on the right of the figure. When we request an operator of assignment or reference of the certain range of the cached data, the part of the data is read directly into Matlab memory so that we could process it as if we deal with a normal Matlab array.

Fast access procedures - an overview

In order to perform the access procedures, we have to figure out the fastest way to do read/write with the range of files kept on disk. The well-known functions such as load, save or even fwrite, fread are not fast enough for that. Therefore, we use the Matlab's memmapfile in order to create a memory map to each of the file.

The memmapfile is incorporated into the subsasgn and subsref functions (bracket operators for reading and writing) which are redefined in CachedNDArray class.

Discreet vs. continious caching

Depending on the CachedNDArray usage, we can define the caching to be discreet (fast) or continious (slow). The figure below helps to differentiate the two:

Alt test

Since the data is split into chunks and each chunk is saved into a separate file, the discreet access would mean only accessing the chunks as they are written within the files as shown on the right, i.e. it is not possible to access a chunk of data that is shared between two files (as shown on the left of the figure). A basic example when we use the discreet caching: given 2D RGB array (color image), filter it so that only blue color is left.

As to the continious caching, the access chunk could be shared between the two consequitive files as shown on the left of the figure. However, it also comes with a price of much slower performance since there will be more copying/writing involved than if we access data dicretely. An example when we use the continious caching: given 2D RGB array (color image), smooth it out so that each output pixel equals to the average sum of its surrounding pixels.

Interface signatures

Constructor

To create a CachedNDArray variable, it is necessary to use the constructor, e.g:

cnda = CachedNDArray(dimensions, broken, ... 
   'type', type, 'var_name', var_name, 'path_cache', work_path, ...
    'nchunks', nchunks, 'fcaching', fcaching, 'fdiscreet', fdiscreet, ...
    'ini_val', ini_val);

where

  • dimensions - is the vector of form [dim_1 dim_2 ... dim_n] that defines the size of each array dimension; it is mandatory parameter.
  • broken - is an integer which defines which dimension will be broken, e.g. for broken = 2 the second dimension dim_2 will be broken; it is mandatory parameter.
  • type - is a string variable, e.g. type = 'double' to define the data type of the array; optional parameter: if omitted, then the default value is set to double
  • var_name - is a string to defined under which name the caching data will be saved, e.g. if var_name = 'tmp', the caching data will be stored in variables {tmp1.dat, tmp2.dat, ... tmpn.dat}. Optional parameter: if omitted, then the default value is set to tmp.
  • work_path is the directory path where the cached *.dat files will be stored, in string format. Optional parameter: if omitted, then the default value is set to cache.
  • nchunks - is an integer which defines the number of chunks the array will be broken into. In case of an automatic breakage: by default the array will be broken into chunks no bigger than 8Gb each. Optional parameter: if omitted, then the default value is set to 0 (it will trigger automatic breakage).
  • fcaching - a caching flag. It is set to -1 by default which triggers automatic decision whether to cache the data or not. It is possible to enforce the data to be cached always - use fcaching = 1, or to always suppress it by fcaching = 0; although it is advised to use the default value for most of the cases: fcaching = -1. Optional parameter: if omitted, then the default value is set to -1.
  • fdiscreet - a flag to define discreet (fast) or continious (slow) caching. If not initialized, the fast caching is used. Optional parameter: if omitted, then the default value is set to 1.
  • ini_val - is an initial value that array will be initialized with, e.g. ini_val = 1.5, ini_val = inf. If this parameter is not provided, by default it is set to 0.
Write operator - subsasgn

An example of how to use the constructor:

cnda = CachedNDArray([100, 1000, 500, 500], 2, ...
    'type', 'single', 'var_name', 'cnda1', 'nchunks', 5, ...
    'fcaching', 1, 'fdiscreet', 0, 'ini_val', inf);

In the above example, the cnda variable is initialized as a CachedNDArray with the next parameters:

  • it is of size [100, 1000, 500, 500]
  • type single
  • with initial value of inf
  • with second broken dimension, i.e. it will be broken into five chunks of size [100, 200, 500, 500]
  • it is mandatory cached - no matter if there is enough memory in the Matlab workspace or not
  • the cached data is saved on disk in current folder under names {cnda11.dat, cnda12.dat, cnda13.dat, cnda14.dat, cnda15.dat}
  • it is assumed that data is processed chunk-wise

The assignment operator has the same signature as when dealing with a normal Matlab array:

cnda(:,1:10,:,:) = chunk;

where
chunk - is the data chunk which we want to write to the cnda array.

Note that the assignment operator must be accompanied by the flush() function (see below) in certain situations for the continious caching access; the flushing is done automatically when dealing with the discreet type.

Read operator - subsref

The reference operator has the same signature as when dealing with a normal Matlab array:

chunk = cnda(:, 80:end, :, :);

where
chunk is the return data chunk which is copied from cnda array

Flushing

cnda.flush() is a function that performs dumping of the data buffer into the corresponding file(-s). The normal data buffer (which is named as a SlidingWindow) corresponds to one file on the disk (size-wise and content-wise). When the SlidingWindow moves from one file to another, it flushes all the changed data to the corresponding file.

The flushing works differently for the discreet and continious access types. When working with discreet array, it is assumed that assignment is done chunk-wise and, therefore, flushing is done every time when we call an assignment operator. In this case, we never need to call the flush() function. Because of this, it is much more efficient to always assign the data chunk as big as the data buffer SlidingWindow (the size of one file representing the CachedNDArray). See the Usage tips for discreet data access for more details and examples.

For the continuous access, the flush() function is called automatically only when the SlidingWindow coordinates change, e.g., when the user reads or writes from the file which is not represented by the SlidingWindow. Because of this specifics, it is necessary to perform the flush() at the very end of the reading / writing loop so that the last chunk that we just wrote to is dumped to the file. Here is a code example:

cnda = CachedNDArray([1000, 100, 100], 'double', 1, 'tmp', 'cache', 4, 1);
for i = 1 : 1000
    line2 = rand([2, 100, 100], 'double');
    cnda(i:i+1, :, :) = line2;
end
cnda.flush(); % flush the last SlidingWindow

Usage tips for discreet data access

For the best performance it is strongly advised to design your program with accordance of the next two points:

  • Discreet data access, i.e. elemnt-wise or line-wise so that each chunk to write is represented by a corresponding file. In case of the discreet data access, the assignment operator should be performed chunk-wise and linearly, i.e. it is useful to keep a variable chunk of a data buffer size and do assignment through this chunk of data.
  • Sequential data access as opposed to random data access (for both discreet and continious access types).

The below code samples may be used as a reference of how it should and should not be implemented:

The chunk-wise sequential assignemnt operator (fastest way):

cnda = CachedNDArray([1000 100 100], 1, 'nchunks', 4, 'fcaching', 1);
for i = 1 : 4
    chunk = rand([250 100 100], 'double');
    cnda((i-1)*250+1 : i*250, :, :) = chunk;
end

As opposed to the next example, when assignment operator is not done chunk-wise in a rather random order. This sample will perform much slower since there will be many more write operators called than in the code sample shown above.

cnda = CachedNDArray([1000 100 100], 1,'nchunks', 4, 'fcaching', 1);
indices = randperm(1000);
for i = 1 : 1000
    chunk = rand([1 100 100], 'double');
    cnda(indices(i), :, :) = chunk; % not chunk-wise and not sequential access
end

The above example could be run much faster if we do assignment chunk-wise and also sort the indices array in order to have the sequential access:

cnda = CachedNDArray([1000 100 100], 1, 'nchunks', 4, 'fcaching', 1);
indices = randperm(1000);
indices_sort = sort(indices);
for i = 1 : 4
    chunk = zeros(250, 100, 100);
    for j = (i-1)*250+1 : i*250
        chunk(indices_sort(j), :, :) = rand([1 100 100], 'double');
    end
    cnda((i-1)*250+1 : i*250, :, :) = chunk;
end

Specifics

It is required that your machine's memory could acommodate 8Gb of memory to hold the maximum size buffer data plus another 8-10Gb for any other side variables and/or processes. You could always change the default parameter of 8Gb to make the computational requirements lower inside the CachedNDArray.m file.

Notes and contact

The class was originally created as a part of a software package cryo3d for fast 3d reconstruction of protein structure from microscopic images.

Contact: 2015 Victoria Rudakova vicrucann(at)gmail(dot)com