/* * \brief Rtc file system * \author Josef Soentgen * \date 2014-08-20 */ /* * Copyright (C) 2014-2017 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU Affero General Public License version 3. */ #ifndef _INCLUDE__VFS__RTC_FILE_SYSTEM_H_ #define _INCLUDE__VFS__RTC_FILE_SYSTEM_H_ /* Genode includes */ #include #include #include namespace Vfs { class Rtc_file_system; } class Vfs::Rtc_file_system : public Single_file_system { private: class Rtc_vfs_handle : public Single_vfs_handle { private: Rtc::Connection &_rtc; public: Rtc_vfs_handle(Directory_service &ds, File_io_service &fs, Genode::Allocator &alloc, Rtc::Connection &rtc) : Single_vfs_handle(ds, fs, alloc, 0), _rtc(rtc) { } /** * Read the current time from the Rtc session * * On each read the current time is queried and afterwards formated * as '%Y-%m-%d %H:%M\n'. */ Read_result read(char *dst, file_size count, file_size &out_count) override { enum { TIMESTAMP_LEN = 17 }; if (seek() >= TIMESTAMP_LEN) { out_count = 0; return READ_OK; } Rtc::Timestamp ts = _rtc.current_time(); char buf[TIMESTAMP_LEN+1]; char *b = buf; unsigned n = Genode::snprintf(buf, sizeof(buf), "%04u-%02u-%02u %02u:%02u\n", ts.year, ts.month, ts.day, ts.hour, ts.minute); n -= seek(); b += seek(); file_size len = count > n ? n : count; Genode::memcpy(dst, b, len); out_count = len; return READ_OK; } Write_result write(char const *, file_size, file_size &) override { return WRITE_ERR_IO; } bool read_ready() override { return true; } }; typedef Genode::Registered Registered_watch_handle; typedef Genode::Registry Watch_handle_registry; Rtc::Connection _rtc; Watch_handle_registry _handle_registry { }; Genode::Io_signal_handler _set_signal_handler; void _handle_set_signal() { _handle_registry.for_each([this] (Registered_watch_handle &handle) { handle.watch_response(); }); } public: Rtc_file_system(Vfs::Env &env, Genode::Xml_node config) : Single_file_system(Node_type::TRANSACTIONAL_FILE, name(), Node_rwx::ro(), config), _rtc(env.env()), _set_signal_handler(env.env().ep(), *this, &Rtc_file_system::_handle_set_signal) { _rtc.set_sigh(_set_signal_handler); } static char const *name() { return "rtc"; } char const *type() override { return "rtc"; } /********************************* ** Directory-service interface ** *********************************/ Open_result open(char const *path, unsigned, Vfs_handle **out_handle, Allocator &alloc) override { if (!_single_file(path)) return OPEN_ERR_UNACCESSIBLE; try { *out_handle = new (alloc) Rtc_vfs_handle(*this, *this, alloc, _rtc); return OPEN_OK; } catch (Genode::Out_of_ram) { return OPEN_ERR_OUT_OF_RAM; } catch (Genode::Out_of_caps) { return OPEN_ERR_OUT_OF_CAPS; } } Stat_result stat(char const *path, Stat &out) override { return Single_file_system::stat(path, out); } Watch_result watch(char const *path, Vfs_watch_handle **handle, Allocator &alloc) override { if (!_single_file(path)) return WATCH_ERR_UNACCESSIBLE; try { Vfs_watch_handle &watch_handle = *new (alloc) Registered_watch_handle(_handle_registry, *this, alloc); *handle = &watch_handle; return WATCH_OK; } catch (Genode::Out_of_ram) { return WATCH_ERR_OUT_OF_RAM; } catch (Genode::Out_of_caps) { return WATCH_ERR_OUT_OF_CAPS; } } }; #endif /* _INCLUDE__VFS__RTC_FILE_SYSTEM_H_ */