summaryrefslogtreecommitdiffstats
path: root/cpukit/libblock
diff options
context:
space:
mode:
authorRalf Kirchner <ralf.kirchner@embedded-brains.de>2012-12-05 10:54:17 +0100
committerSebastian Huber <sebastian.huber@embedded-brains.de>2012-12-05 15:26:47 +0100
commit7fab7fc0cf00718dd917bc53efea75a702663498 (patch)
tree6ab47b347d256054e80a9d3535bbf9287783cd75 /cpukit/libblock
parentrtems misc: Clean up Doxygen GCI Task #4 (diff)
downloadrtems-7fab7fc0cf00718dd917bc53efea75a702663498.tar.bz2
libblock: Add sparse disk
Diffstat (limited to 'cpukit/libblock')
-rw-r--r--cpukit/libblock/Makefile.am1
-rw-r--r--cpukit/libblock/include/rtems/sparse-disk.h137
-rw-r--r--cpukit/libblock/src/sparse-disk.c387
3 files changed, 525 insertions, 0 deletions
diff --git a/cpukit/libblock/Makefile.am b/cpukit/libblock/Makefile.am
index d07eb5e177..cafd1fcba0 100644
--- a/cpukit/libblock/Makefile.am
+++ b/cpukit/libblock/Makefile.am
@@ -31,6 +31,7 @@ libblock_a_SOURCES = src/bdbuf.c \
src/media-server.c \
src/media-desc.c \
src/media-dev-ident.c \
+ src/sparse-disk.c \
include/rtems/bdbuf.h include/rtems/blkdev.h \
include/rtems/diskdevs.h include/rtems/flashdisk.h \
include/rtems/ramdisk.h include/rtems/nvdisk.h include/rtems/nvdisk-sram.h \
diff --git a/cpukit/libblock/include/rtems/sparse-disk.h b/cpukit/libblock/include/rtems/sparse-disk.h
new file mode 100644
index 0000000000..3dc806b407
--- /dev/null
+++ b/cpukit/libblock/include/rtems/sparse-disk.h
@@ -0,0 +1,137 @@
+/**
+ * @file
+ *
+ * @ingroup rtems_sparse_disk
+ *
+ * @brief Sparse disk block device API.
+ */
+
+/*
+ * Copyright (c) 2012 embedded brains GmbH. All rights reserved.
+ *
+ * embedded brains GmbH
+ * Obere Lagerstr. 30
+ * 82178 Puchheim
+ * Germany
+ * <rtems@embedded-brains.de>
+ *
+ * The license and distribution terms for this file may be
+ * found in the file LICENSE in this distribution or at
+ * http://www.rtems.com/license/LICENSE.
+ */
+
+#ifndef SPARSE_DISK_H
+#define SPARSE_DISK_H
+
+#include <stddef.h>
+#include <stdint.h>
+#include <rtems.h>
+#include <rtems/diskdevs.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif /* __cplusplus */
+
+/**
+ * @defgroup rtems_sparse_disk Sparse Disk Device
+ *
+ * @ingroup rtems_blkdev
+ *
+ * @{
+ */
+
+typedef struct {
+ rtems_blkdev_bnum block;
+ void *data;
+} rtems_sparse_disk_key;
+
+typedef struct rtems_sparse_disk rtems_sparse_disk;
+
+typedef void (*rtems_sparse_disk_delete_handler)(rtems_sparse_disk *sparse_disk);
+
+struct rtems_sparse_disk {
+ rtems_id mutex;
+ rtems_blkdev_bnum blocks_with_buffer;
+ size_t used_count;
+ uint32_t media_block_size;
+ rtems_sparse_disk_delete_handler delete_handler;
+ uint8_t fill_pattern;
+ rtems_sparse_disk_key *key_table;
+};
+
+/**
+ * @brief Creates and registers a sparse disk.
+ *
+ * @param[in] device_file_name The device file name path.
+ * @param[in] media_block_size The media block size in bytes.
+ * @param[in] blocks_with_buffer Blocks of the device with a buffer. Other
+ * blocks can store only fill pattern value bytes.
+ * @param[in] block_count The media block count of the device. It is the sum
+ * of blocks with buffer and blocks that contain only fill pattern value bytes.
+ * @param[in] fill_pattern The fill pattern specifies the byte value of blocks
+ * without a buffer. It is also the initial value for blocks with a buffer.
+ *
+ * @retval RTEMS_SUCCESSFUL Successful operation.
+ * @retval RTEMS_INVALID_NUMBER Media block size or media block count is not
+ * positive. The blocks with buffer count is greater than the media block count.
+ * @retval RTEMS_NO_MEMORY Not enough memory.
+ * @retval RTEMS_TOO_MANY Cannot create semaphore.
+ * @retval RTEMS_UNSATISFIED Cannot create generic device node.
+ *
+ * @see rtems_sparse_disk_register().
+ */
+rtems_status_code rtems_sparse_disk_create_and_register(
+ const char *device_file_name,
+ uint32_t media_block_size,
+ rtems_blkdev_bnum blocks_with_buffer,
+ rtems_blkdev_bnum media_block_count,
+ uint8_t fill_pattern
+);
+
+/**
+ * @brief Frees a sparse disk.
+ *
+ * Calls free() on the sparse disk pointer.
+ */
+void rtems_sparse_disk_free( rtems_sparse_disk *sparse_disk );
+
+/**
+ * @brief Initializes and registers a sparse disk.
+ *
+ * This will create one semaphore for mutual exclusion.
+ *
+ * @param[in] device_file_name The device file name path.
+ * @param[in, out] sparse_disk The sparse disk.
+ * @param[in] media_block_size The media block size in bytes.
+ * @param[in] blocks_with_buffer Blocks of the device with a buffer. Other
+ * blocks can store only fill pattern value bytes.
+ * @param[in] block_count The media block count of the device. It is the sum
+ * of blocks with buffer and blocks that contain only fill pattern value bytes.
+ * @param[in] fill_pattern The fill pattern specifies the byte value of blocks
+ * without a buffer. It is also the initial value for blocks with a buffer.
+ * @param[in] sparse_disk_delete The sparse disk delete handler.
+ *
+ * @retval RTEMS_SUCCESSFUL Successful operation.
+ * @retval RTEMS_INVALID_NUMBER Media block size or media block count is not
+ * positive. The blocks with buffer count is greater than the media block count.
+ * @retval RTEMS_INVALID_ADDRESS Invalid sparse disk address.
+ * @retval RTEMS_TOO_MANY Cannot create semaphore.
+ * @retval RTEMS_UNSATISFIED Cannot create generic device node.
+ */
+rtems_status_code rtems_sparse_disk_register(
+ const char *device_file_name,
+ rtems_sparse_disk *sparse_disk,
+ uint32_t media_block_size,
+ rtems_blkdev_bnum blocks_with_buffer,
+ rtems_blkdev_bnum media_block_count,
+ uint8_t fill_pattern,
+ rtems_sparse_disk_delete_handler sparse_disk_delete
+);
+
+/** @} */
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* SPARSE_DISK_H */
diff --git a/cpukit/libblock/src/sparse-disk.c b/cpukit/libblock/src/sparse-disk.c
new file mode 100644
index 0000000000..987ab6d806
--- /dev/null
+++ b/cpukit/libblock/src/sparse-disk.c
@@ -0,0 +1,387 @@
+/**
+ * @file
+ *
+ * @ingroup rtems_sparse_disk
+ *
+ * @brief Sparse disk block device implementation.
+ */
+
+/*
+ * Copyright (c) 2012 embedded brains GmbH. All rights reserved.
+ *
+ * embedded brains GmbH
+ * Obere Lagerstr. 30
+ * 82178 Puchheim
+ * Germany
+ * <rtems@embedded-brains.de>
+ *
+ * The license and distribution terms for this file may be
+ * found in the file LICENSE in this distribution or at
+ * http://www.rtems.com/license/LICENSE.
+ */
+
+#include <stdlib.h>
+#include <errno.h>
+#include <string.h>
+
+#include <rtems.h>
+#include <rtems/blkdev.h>
+#include <rtems/fatal.h>
+
+#include "rtems/sparse-disk.h"
+
+/*
+ * Allocate RAM for sparse disk
+ */
+static rtems_sparse_disk *sparse_disk_allocate(
+ const uint32_t media_block_size,
+ const rtems_blkdev_bnum blocks_with_buffer )
+{
+ size_t const key_table_size = blocks_with_buffer
+ * sizeof( rtems_sparse_disk_key );
+ size_t const data_size = blocks_with_buffer * media_block_size;
+ size_t const alloc_size = sizeof( rtems_sparse_disk )
+ + key_table_size + data_size;
+
+ rtems_sparse_disk *const sd = (rtems_sparse_disk *) malloc(
+ alloc_size );
+
+ return sd;
+}
+
+/*
+ * Initialize sparse disk data
+ */
+static rtems_status_code sparse_disk_initialize( rtems_sparse_disk *sd,
+ const uint32_t media_block_size,
+ const rtems_blkdev_bnum blocks_with_buffer,
+ const rtems_sparse_disk_delete_handler sparse_disk_delete,
+ const uint8_t fill_pattern )
+{
+ rtems_status_code sc;
+ rtems_blkdev_bnum i;
+
+ if ( NULL == sd )
+ return RTEMS_INVALID_ADDRESS;
+
+ uint8_t *data = (uint8_t *) sd;
+ size_t const key_table_size = blocks_with_buffer
+ * sizeof( rtems_sparse_disk_key );
+ size_t const data_size = blocks_with_buffer * media_block_size;
+
+ memset( data, 0, sizeof( rtems_sparse_disk ) + key_table_size );
+
+ sd->fill_pattern = fill_pattern;
+ memset( (uint8_t *) ( data + sizeof( rtems_sparse_disk ) + key_table_size ),
+ sd->fill_pattern,
+ data_size );
+
+ sd->delete_handler = sparse_disk_delete;
+
+ sc = rtems_semaphore_create(
+ rtems_build_name( 'S', 'P', 'A', 'R' ),
+ 1,
+ RTEMS_PRIORITY | RTEMS_BINARY_SEMAPHORE | RTEMS_INHERIT_PRIORITY,
+ 0,
+ &sd->mutex
+ );
+
+ if ( sc != RTEMS_SUCCESSFUL ) {
+ return sc;
+ }
+
+ data += sizeof( rtems_sparse_disk );
+
+ sd->blocks_with_buffer = blocks_with_buffer;
+ sd->key_table = (rtems_sparse_disk_key *) data;
+
+ data += key_table_size;
+
+ for ( i = 0; i < blocks_with_buffer; ++i, data += media_block_size ) {
+ sd->key_table[i].data = data;
+ }
+
+ sd->media_block_size = media_block_size;
+ return RTEMS_SUCCESSFUL;
+}
+
+/*
+ * Block comparison
+ */
+static int sparse_disk_compare( const void *aa, const void *bb )
+{
+ const rtems_sparse_disk_key *a = aa;
+ const rtems_sparse_disk_key *b = bb;
+
+ if ( a->block < b->block ) {
+ return -1;
+ } else if ( a->block == b->block ) {
+ return 0;
+ } else {
+ return 1;
+ }
+}
+
+static rtems_sparse_disk_key *sparse_disk_get_new_block(
+ rtems_sparse_disk *sparse_disk,
+ const rtems_blkdev_bnum block )
+{
+ rtems_sparse_disk_key *key;
+
+ if ( sparse_disk->used_count < sparse_disk->blocks_with_buffer ) {
+ key = &sparse_disk->key_table[sparse_disk->used_count];
+ key->block = block;
+ ++sparse_disk->used_count;
+ qsort( sparse_disk->key_table, sparse_disk->used_count,
+ sizeof( rtems_sparse_disk_key ), sparse_disk_compare );
+ } else
+ return NULL;
+
+ return key;
+}
+
+static int sparse_disk_read_block(
+ const rtems_sparse_disk *sparse_disk,
+ const rtems_blkdev_bnum block,
+ uint8_t *buffer,
+ const size_t buffer_size )
+{
+ rtems_sparse_disk_key *key;
+ rtems_sparse_disk_key block_key = {
+ .block = block,
+ .data = NULL
+ };
+ size_t bytes_to_copy = sparse_disk->media_block_size;
+
+ if ( buffer_size < bytes_to_copy )
+ bytes_to_copy = buffer_size;
+
+ key = bsearch(
+ &block_key,
+ sparse_disk->key_table,
+ sparse_disk->used_count,
+ sizeof( rtems_sparse_disk_key ),
+ sparse_disk_compare
+ );
+
+ if ( NULL != key )
+ memcpy( buffer, key->data, bytes_to_copy );
+ else
+ memset( buffer, sparse_disk->fill_pattern, buffer_size );
+
+ return bytes_to_copy;
+}
+
+static int sparse_disk_write_block(
+ rtems_sparse_disk *sparse_disk,
+ const rtems_blkdev_bnum block,
+ const uint8_t *buffer,
+ const size_t buffer_size )
+{
+ unsigned int i;
+ bool block_needs_writing = false;
+ rtems_sparse_disk_key *key;
+ rtems_sparse_disk_key block_key = {
+ .block = block,
+ .data = NULL
+ };
+ size_t bytes_to_copy = sparse_disk->media_block_size;
+
+ if ( buffer_size < bytes_to_copy )
+ bytes_to_copy = buffer_size;
+
+ /* we only need to write the block if it is different from the fill pattern.
+ * If the read method does not find a block it will deliver the fill pattern anyway.
+ */
+
+ key = bsearch(
+ &block_key,
+ sparse_disk->key_table,
+ sparse_disk->used_count,
+ sizeof( rtems_sparse_disk_key ),
+ sparse_disk_compare
+ );
+
+ if ( NULL == key ) {
+ for ( i = 0; ( !block_needs_writing ) && ( i < bytes_to_copy ); ++i ) {
+ if ( buffer[i] != sparse_disk->fill_pattern )
+ block_needs_writing = true;
+ }
+
+ if ( block_needs_writing ) {
+ key = sparse_disk_get_new_block( sparse_disk, block );
+ }
+ }
+
+ if ( NULL != key )
+ memcpy( key->data, buffer, bytes_to_copy );
+ else if ( block_needs_writing )
+ return -1;
+
+ return bytes_to_copy;
+}
+
+/*
+ * Read/write handling
+ */
+static int sparse_disk_read_write(
+ rtems_sparse_disk *sparse_disk,
+ rtems_blkdev_request *req,
+ const bool read )
+{
+ int rv = 0;
+ uint32_t req_buffer;
+ rtems_blkdev_sg_buffer *scatter_gather;
+ rtems_blkdev_bnum block;
+ uint8_t *buff;
+ size_t buff_size;
+ unsigned int bytes_handled;
+
+ rtems_semaphore_obtain( sparse_disk->mutex, RTEMS_WAIT, RTEMS_NO_TIMEOUT );
+
+ for ( req_buffer = 0;
+ ( 0 <= rv ) && ( req_buffer < req->bufnum );
+ ++req_buffer ) {
+ scatter_gather = &req->bufs[req_buffer];
+
+ bytes_handled = 0;
+ buff = (uint8_t *) scatter_gather->buffer;
+ block = scatter_gather->block;
+ buff_size = scatter_gather->length;
+
+ while ( ( 0 <= rv ) && ( 0 < buff_size ) ) {
+ if ( read )
+ rv = sparse_disk_read_block( sparse_disk,
+ block,
+ &buff[bytes_handled],
+ buff_size );
+ else
+ rv = sparse_disk_write_block( sparse_disk,
+ block,
+ &buff[bytes_handled],
+ buff_size );
+
+ ++block;
+ bytes_handled += rv;
+ buff_size -= rv;
+ }
+ }
+
+ rtems_semaphore_release( sparse_disk->mutex );
+
+ if ( 0 > rv )
+ rtems_blkdev_request_done( req, RTEMS_IO_ERROR );
+ else
+ rtems_blkdev_request_done( req, RTEMS_SUCCESSFUL );
+
+ return 0;
+}
+
+/*
+ * ioctl handler to be passed to the block device handler
+ */
+static int sparse_disk_ioctl( rtems_disk_device *dd, uint32_t req, void *argp )
+{
+ rtems_status_code sc;
+ rtems_sparse_disk *sd = rtems_disk_get_driver_data( dd );
+
+ if ( RTEMS_BLKIO_REQUEST == req ) {
+ rtems_blkdev_request *r = argp;
+
+ switch ( r->req ) {
+ case RTEMS_BLKDEV_REQ_READ:
+ case RTEMS_BLKDEV_REQ_WRITE:
+ return sparse_disk_read_write( sd, r, r->req == RTEMS_BLKDEV_REQ_READ );
+ default:
+ break;
+ }
+ } else if ( RTEMS_BLKIO_DELETED == req ) {
+ sc = rtems_semaphore_delete( sd->mutex );
+
+ if ( RTEMS_SUCCESSFUL != sc )
+ rtems_fatal_error_occurred( 0xdeadbeef );
+
+ sd->mutex = RTEMS_ID_NONE;
+
+ if ( NULL != sd->delete_handler )
+ ( *sd->delete_handler )( sd );
+
+ return 0;
+ } else {
+ return rtems_blkdev_ioctl( dd, req, argp );
+ }
+
+ errno = EINVAL;
+ return -1;
+}
+
+void rtems_sparse_disk_free( rtems_sparse_disk *sd )
+{
+ free( sd );
+}
+
+rtems_status_code rtems_sparse_disk_create_and_register(
+ const char *device_file_name,
+ uint32_t media_block_size,
+ rtems_blkdev_bnum blocks_with_buffer,
+ rtems_blkdev_bnum media_block_count,
+ uint8_t fill_pattern )
+{
+ rtems_status_code sc = RTEMS_SUCCESSFUL;
+ rtems_sparse_disk *sparse_disk = sparse_disk_allocate(
+ media_block_size,
+ blocks_with_buffer
+ );
+
+ if ( sparse_disk != NULL ) {
+ sc = rtems_sparse_disk_register(
+ device_file_name,
+ sparse_disk,
+ media_block_size,
+ blocks_with_buffer,
+ media_block_count,
+ fill_pattern,
+ rtems_sparse_disk_free
+ );
+ } else {
+ sc = RTEMS_NO_MEMORY;
+ }
+
+ return sc;
+}
+
+rtems_status_code rtems_sparse_disk_register(
+ const char *device_file_name,
+ rtems_sparse_disk *sparse_disk,
+ uint32_t media_block_size,
+ rtems_blkdev_bnum blocks_with_buffer,
+ rtems_blkdev_bnum media_block_count,
+ uint8_t fill_pattern,
+ rtems_sparse_disk_delete_handler sparse_disk_delete )
+{
+ rtems_status_code sc;
+
+ if ( blocks_with_buffer <= media_block_count ) {
+ sc = sparse_disk_initialize(
+ sparse_disk,
+ media_block_size,
+ blocks_with_buffer,
+ sparse_disk_delete,
+ fill_pattern
+ );
+
+ if ( RTEMS_SUCCESSFUL == sc ) {
+ sc = rtems_blkdev_create(
+ device_file_name,
+ media_block_size,
+ media_block_count,
+ sparse_disk_ioctl,
+ sparse_disk
+ );
+ }
+ } else {
+ sc = RTEMS_INVALID_NUMBER;
+ }
+
+ return sc;
+}