/* * \brief Service that provides files of a TAR archive as ROM sessions * \author Martin stein * \author Norman Feske * \date 2010-04-21 */ /* * Copyright (C) 2010-2013 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU General Public License version 2. */ /* Genode includes */ #include #include #include #include #include #include #include #include #include /** * A 'Rom_session_component' exports a single file of the tar archive */ class Rom_session_component : public Genode::Rpc_object { private: const char *_tar_addr, *_filename, *_file_addr; Genode::size_t _file_size, _tar_size; Genode::Ram_dataspace_capability _file_ds; enum { /* length of on data block in tar */ _BLOCK_LEN = 512, /* length of the header field "file-size" in tar */ _FIELD_SIZE_LEN = 124 }; /** * Copy file content into dataspace * * \param dst destination dataspace */ void _copy_content_to_dataspace(Genode::Dataspace_capability dst) { using namespace Genode; /* map dataspace locally */ char *dst_addr = env()->rm_session()->attach(dst); Dataspace_client dst_client(dst); /* copy content */ size_t dst_ds_size = Dataspace_client(dst).size(); size_t bytes_to_copy = min(_file_size, dst_ds_size); memcpy(dst_addr, _file_addr, bytes_to_copy); /* unmap dataspace */ env()->rm_session()->detach(dst_addr); } /** * Initialize dataspace containing the content of the archived file */ Genode::Ram_dataspace_capability _init_file_ds() { bool file_found = false; /* measure size of archive in blocks */ unsigned block_id = 0, block_cnt = _tar_size/_BLOCK_LEN; /* scan metablocks of archive */ while (block_id < block_cnt) { unsigned long file_size = 0; Genode::ascii_to_unsigned(_tar_addr + block_id*_BLOCK_LEN + _FIELD_SIZE_LEN, file_size, 8); /* get name of tar record */ char const *record_filename = _tar_addr + block_id*_BLOCK_LEN; /* skip leading dot of path if present */ if (record_filename[0] == '.' && record_filename[1] == '/') record_filename++; /* get infos about current file */ if (Genode::strcmp(_filename, record_filename) == 0) { _file_size = file_size; _file_addr = _tar_addr + (block_id+1) * _BLOCK_LEN; file_found = true; break; } /* some datablocks */ /* one metablock */ block_id = block_id + (file_size / _BLOCK_LEN) + 1; /* round up */ if (file_size % _BLOCK_LEN != 0) block_id++; /* check for end of tar archive */ if (block_id*_BLOCK_LEN >= _tar_size) break; /* lookout for empty eof-blocks */ if (*(_tar_addr + (block_id*_BLOCK_LEN)) == 0x00) if (*(_tar_addr + (block_id*_BLOCK_LEN + 1)) == 0x00) break; } if (!file_found) { PERR("couldn't find file '%s', empty result", _filename); return Genode::Ram_dataspace_capability(); } /* try to allocate memory for file */ Genode::Ram_dataspace_capability file_ds; try { file_ds = Genode::env()->ram_session()->alloc(_file_size); /* get content of file copied into dataspace and return */ _copy_content_to_dataspace(file_ds); } catch (...) { PERR("couldn't allocate memory for file, empty result\n"); return file_ds; } return file_ds; } public: /** * Constructor scans and seeks to file * * \param tar_addr local address to tar archive * \param tar_size size of tar archive in bytes * \param filename name of the requested file */ Rom_session_component(const char *tar_addr, unsigned tar_size, const char *filename) : _tar_addr(tar_addr), _filename(filename), _file_addr(0), _file_size(0), _tar_size(tar_size), _file_ds(_init_file_ds()) { if (!_file_ds.valid()) throw Genode::Root::Invalid_args(); } /** * Destructor */ ~Rom_session_component() { Genode::env()->ram_session()->free(_file_ds); } /** * Return dataspace with content of file */ Genode::Rom_dataspace_capability dataspace() { Genode::Dataspace_capability ds = _file_ds; return Genode::static_cap_cast(ds); } void sigh(Genode::Signal_context_capability) { } }; class Rom_root : public Genode::Root_component { private: char *_tar_addr; unsigned _tar_size; Rom_session_component *_create_session(const char *args) { enum { FILENAME_MAX_LEN = 128 }; char filename[FILENAME_MAX_LEN]; Genode::Arg_string::find_arg(args, "filename").string(filename, sizeof(filename), ""); PINF("connection for file '%s' requested\n", filename); /* create new session for the requested file */ return new (md_alloc()) Rom_session_component(_tar_addr, _tar_size, filename); } public: /** * Constructor * * \param entrypoint entrypoint to be used for ROM sessions * \param md_alloc meta-data allocator used for ROM sessions * \param tar_base local address of tar archive * \param tar_size size of tar archive in bytes */ Rom_root(Genode::Rpc_entrypoint *entrypoint, Genode::Allocator *md_alloc, char *tar_addr, Genode::size_t tar_size) : Genode::Root_component(entrypoint, md_alloc), _tar_addr(tar_addr), _tar_size(tar_size) { } }; using namespace Genode; int main(void) { /* read name of tar archive from config */ enum { TAR_FILENAME_MAX_LEN = 64 }; static char tar_filename[TAR_FILENAME_MAX_LEN]; try { Xml_node archive_node = config()->xml_node().sub_node("archive"); archive_node.attribute("name").value(tar_filename, sizeof(tar_filename)); } catch (...) { PERR("Could not read 'filename' argument from config"); return -1; } /* obtain dataspace of tar archive from ROM service */ static char *tar_base = 0; static size_t tar_size = 0; try { static Rom_connection tar_rom(tar_filename); tar_base = env()->rm_session()->attach(tar_rom.dataspace()); tar_size = Dataspace_client(tar_rom.dataspace()).size(); } catch (...) { PERR("Could not obtain tar archive from ROM service"); return -2; } PINF("using tar archive '%s' with size %zd", tar_filename, tar_size); /* connection to capability service needed to create capabilities */ static Cap_connection cap; /* creation of the entrypoint and the root interface */ static Sliced_heap sliced_heap(env()->ram_session(), env()->rm_session()); enum { STACK_SIZE = 8*1024 }; static Rpc_entrypoint ep(&cap, STACK_SIZE, "tar_rom_ep"); static Rom_root rom_root(&ep, &sliced_heap, tar_base, tar_size); /* announce server*/ env()->parent()->announce(ep.manage(&rom_root)); /* wait for activation through client */ sleep_forever(); return 0; }