summaryrefslogtreecommitdiff
path: root/ext2fs
diff options
context:
space:
mode:
Diffstat (limited to 'ext2fs')
-rw-r--r--ext2fs/storeinfo.c229
1 files changed, 129 insertions, 100 deletions
diff --git a/ext2fs/storeinfo.c b/ext2fs/storeinfo.c
index 7d6bdd23..419809c9 100644
--- a/ext2fs/storeinfo.c
+++ b/ext2fs/storeinfo.c
@@ -18,128 +18,157 @@
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
-#include <netinet/in.h>
+#include <string.h>
+#include <netinet/in.h> /* htonl */
#include "ext2fs.h"
error_t
-diskfs_S_file_get_storage_info (struct protid *cred, int *class,
- off_t **runs, unsigned *runs_len,
- size_t *block_size,
- char *dev_name, mach_port_t *dev_port,
- mach_msg_type_name_t *dev_port_type,
- char **misc, unsigned *misc_len,
- int *flags)
+diskfs_S_file_get_storage_info (struct protid *cred,
+ mach_port_t **ports,
+ mach_msg_type_name_t *ports_type,
+ mach_msg_type_number_t *num_ports,
+ int **ints, mach_msg_type_number_t *num_ints,
+ off_t **offsets,
+ mach_msg_type_number_t *num_offsets,
+ char **data, mach_msg_type_number_t *data_len)
{
error_t err = 0;
- block_t index = 0;
- unsigned num_fs_blocks;
- unsigned runs_alloced = 0;
- off_t *run = 0;
- struct node *node = cred->po->np;
-
- *misc_len = sizeof (long) * 4;
- err = vm_allocate (mach_task_self (), (vm_address_t *)misc, *misc_len, 1);
- if (err)
- return err;
+ size_t name_len =
+ (diskfs_device_name && *diskfs_device_name)
+ ? strlen (diskfs_device_name) + 1 : 0;
+ /* True when we've allocated memory for the corresponding vector. */
+ int al_ports = 0, al_ints = 0, al_offsets = 0, al_data = 0;
+
+ if (! cred)
+ return EOPNOTSUPP;
+
+#define ENSURE_MEM(v, vl, alp, num) \
+ if (!err && *vl < num) \
+ { \
+ err = vm_allocate (mach_task_self (), \
+ (vm_address_t *)v, num * sizeof (**v), 1); \
+ if (! err) \
+ { \
+ *vl = num; \
+ alp = 1; \
+ } \
+ }
+
+ /* Two longs. */
+#define MISC_LEN (sizeof (long) * 2)
- mutex_lock (&node->lock);
+ ENSURE_MEM (ports, num_ports, al_ports, 1);
+ ENSURE_MEM (ints, num_ints, al_ints, 6);
+ ENSURE_MEM (data, data_len, al_data, name_len + MISC_LEN);
+ /* OFFSETS is more complex, and done below. */
- num_fs_blocks = node->dn_stat.st_blocks >> log2_stat_blocks_per_fs_block;
- while (num_fs_blocks > 0)
+ if (! err)
{
- block_t block;
+ block_t index = 0;
+ unsigned num_fs_blocks;
+ off_t *run = *num_offsets ? *offsets : 0;
+ struct node *node = cred->po->np;
- err = ext2_getblk (node, index++, 0, &block);
- if (err == EINVAL)
- /* Either a hole, or past the end of the file. */
- {
- block = 0;
- err = 0;
- }
- if (err)
- goto fail;
-
- block <<= log2_dev_blocks_per_fs_block;
- if (!run
- || ((block && run[0] >= 0) /* Neither is a hole and... */
- ? (block != run[0] + run[1]) /* ... BLOCK doesn't follow RUN */
- : (block || run[0] >= 0))) /* or ... one is, but not both */
- /* Add a new run. */
+ mutex_lock (&node->lock);
+
+ num_fs_blocks = node->dn_stat.st_blocks >> log2_stat_blocks_per_fs_block;
+ while (num_fs_blocks > 0)
{
- if (run)
- /* There are already some runs. */
+ block_t block;
+
+ err = ext2_getblk (node, index++, 0, &block);
+ if (err == EINVAL)
+ /* Either a hole, or past the end of the file. */
{
- run += 2;
- if (run >= *runs + runs_alloced)
- /* Add a new page to the end of the existing RUNS array. */
- {
- err = vm_allocate (mach_task_self (),
- (vm_address_t *)&run, vm_page_size, 0);
- if (err)
- goto fail;
- runs_alloced += vm_page_size / sizeof (off_t);
- }
+ block = 0;
+ err = 0;
}
- else
- /* Allocate the RUNS array for the first time. */
+ else if (err)
+ break;
+
+ block <<= log2_dev_blocks_per_fs_block;
+ if (!run
+ || ((block && run[0] >= 0) /* Neither is a hole and... */
+ ? (block != run[0] + run[1]) /* BLOCK doesn't follow RUN */
+ : (block || run[0] >= 0))) /* or one is, but not both */
+ /* Add a new run. */
{
- err = vm_allocate (mach_task_self (),
- (vm_address_t *)runs, vm_page_size, 1);
- if (err)
- goto fail;
- runs_alloced = vm_page_size / sizeof (off_t);
- run = *runs;
+ run += 2;
+ if (!run || run >= *offsets + *num_offsets)
+ if (al_offsets)
+ /* We've already allocated space for offsets; add a new
+ page to the end of it. */
+ {
+ err =
+ vm_allocate (mach_task_self (),
+ (vm_address_t *)&run, vm_page_size, 0);
+ if (err)
+ break;
+ *num_offsets += vm_page_size / sizeof (off_t);
+ }
+ else
+ /* We've run out the space passed for inline offsets by
+ the caller, so allocate our own memory and copy
+ anything we've already stored. */
+ {
+ off_t *old = *offsets;
+ size_t old_len = *num_offsets;
+ err =
+ vm_allocate (mach_task_self (),
+ (vm_address_t *)offsets,
+ old_len * sizeof (off_t) + vm_page_size, 1);
+ if (err)
+ break;
+ if (old_len)
+ bcopy (old, *offsets, old_len * sizeof (off_t));
+ *num_offsets = old_len + vm_page_size / sizeof (off_t);
+ run = *offsets;
+ al_offsets = 1;
+ }
+
+ run[0] = block ?: -1; /* -1 means a hole in OFFSETS */
+ run[1] = 0; /* will get extended just below */
}
- run[0] = block ?: -1; /* -1 means a hole in RUNS */
- run[1] = 0; /* will get extended just below */
- }
-
- /* Increase the size of the current run by one filesystem block. */
- run[1] += 1 << log2_dev_blocks_per_fs_block;
+ /* Increase the size of the current run by one filesystem block. */
+ run[1] += 1 << log2_dev_blocks_per_fs_block;
- num_fs_blocks--;
- }
+ num_fs_blocks--;
+ }
- if (run)
- {
- if (run[0] >= 0)
- /* Include the current run, as long as it's not a hole. */
- run += 2;
- else if ((off_t *)trunc_page (run) == run)
- /* We allocated just *one* too many pages -- the last run is a hole. */
- vm_deallocate (mach_task_self (), (vm_address_t)run, vm_page_size);
- *runs_len = run - *runs;
+ /* Fill in PORTS. Root gets device port, everyone else, nothing. */
+ (*ports)[0] = diskfs_isuid (0, cred) ? diskfs_device : MACH_PORT_NULL;
+ *ports_type = MACH_MSG_TYPE_COPY_SEND;
+
+ /* Fill in INTS. */
+ (*ints)[0] = STORAGE_DEVICE; /* type */
+ (*ints)[1] = 0; /* flags */
+ (*ints)[2] = diskfs_device_block_size; /* block size */
+ (*ints)[3] = (run - *offsets) / 2; /* num runs */
+ (*ints)[4] = name_len;
+ (*ints)[5] = MISC_LEN;
+
+ /* Fill in DATA. */
+ if (name_len)
+ strcpy (*data, diskfs_device_name);
+ /* The following must be kept in sync with MISC_LEN. */
+ ((long *)(*data + name_len))[0] = htonl (node->cache_id);
+ ((long *)(*data + name_len))[1] =
+ htonl (dino (node->cache_id)->i_translator);
+
+ mutex_unlock (&node->lock);
}
- else
- *runs_len = 0;
-
- ((long *)*misc)[0] = htonl (node->cache_id);
- ((long *)*misc)[1] = htonl (dino (node->cache_id)->i_translator);
-
- *class = STORAGE_DEVICE;
- *flags = 0;
-
- *block_size = diskfs_device_block_size;
-
- strcpy (dev_name, diskfs_device_name);
-
- if (diskfs_isuid (0, cred))
- *dev_port = diskfs_device;
- else
- *dev_port = MACH_PORT_NULL;
- *dev_port_type = MACH_MSG_TYPE_COPY_SEND;
-
- fail:
- mutex_unlock (&node->lock);
if (err)
{
- if (*runs_len > 0)
- vm_deallocate (mach_task_self (), (vm_address_t)*runs,
- runs_alloced * sizeof (off_t));
- vm_deallocate (mach_task_self (), (vm_address_t)*misc, *misc_len);
+#define DISCARD_MEM(v, vl, alp) \
+ if (alp) \
+ vm_deallocate (mach_task_self (), (vm_address_t)*v, *vl * sizeof **v);
+ DISCARD_MEM (ports, num_ports, al_ports);
+ DISCARD_MEM (ints, num_ints, al_ints);
+ DISCARD_MEM (offsets, num_offsets, al_offsets);
+ DISCARD_MEM (data, data_len, al_data);
}
return err;