diff --git a/daemon/README.rst b/daemon/README.rst
new file mode 100644
index 0000000000000000000000000000000000000000..1c625877b0d43950806b26e1cc2492bfdc70c359
--- /dev/null
+++ b/daemon/README.rst
@@ -0,0 +1,326 @@
+************************
+Knot DNS Resolver daemon 
+************************
+
+Requirements
+============
+
+* libuv_ 1.0+ (a multi-platform support library with a focus on asynchronous I/O)
+* Lua_ 5.1+ (embeddable scripting language, LuaJIT_ is preferred)
+
+Running
+=======
+
+There is a separate resolver library in the `lib` directory, and a minimalistic daemon in
+the `daemon` directory.
+
+.. code-block:: bash
+
+	$ ./daemon/kresolved -h
+
+Interacting with the daemon
+---------------------------
+
+The daemon features a CLI interface if launched interactively, type ``help`` to see the list of available commands.
+You can load modules this way and use their properties to get information about statistics and such.
+
+.. code-block:: bash
+
+	$ kresolved /var/run/knot-resolver
+	[system] started in interactive mode, type 'help()'
+	> cache.count()
+	53
+
+.. role:: lua(code)
+   :language: lua
+
+Configuration
+=============
+
+.. contents::
+   :depth: 2
+   :local:
+
+In it's simplest form it requires just a working directory in which it can set up persistent files like
+cache and the process state. If you don't provide the working directory by parameter, it is going to make itself
+comfortable in the current working directory.
+
+.. code-block:: sh
+
+	$ kresolved /var/run/kresolved
+
+And you're good to go for most use cases! If you want to use modules or configure daemon behavior, read on.
+
+There are several choices on how you can configure the daemon, a RPC interface a CLI and a configuration file.
+Fortunately all share common syntax and are transparent to each other, e.g. changes made during the runtime are kept
+in the redo log and are immediately visible.
+
+.. warning:: Redo log is not yet implemented, changes are visible during the process lifetime only.
+
+Configuration example
+---------------------
+.. code-block:: lua
+
+	-- 10MB cache
+	cache.open(10*MB)
+	-- static hints
+	modules = {
+		hints = true,
+		cachectl = true
+	}
+	-- interfaces
+	net.listen('127.0.0.1')
+
+Configuration syntax
+--------------------
+
+The configuration is kept in the ``config`` file in the daemon working directory, and it's going to get loaded automatically.
+If there isn't one, the daemon is going to start with sane defaults, listening on `localhost`.
+The syntax for options is like follows: ``group.option = value`` or ``group.action(parameters)``.
+You can also comment using a ``--`` prefix.
+
+A simple example would be to load static hints.
+
+.. code-block:: lua
+
+	modules = {
+		cachectl = true -- no configuration
+	}
+
+If the module accepts accepts configuration, you can provide a table.
+The syntax for table is ``{ key1 = value, key2 = value }``, and it represents the unpacked `JSON-encoded`_ string, that
+the modules use as the :ref:`input configuration <mod-properties>`.
+
+.. code-block:: lua
+
+	modules = {
+		cachectl = true,
+		hints = {
+			file = '/etc/hosts'
+		}
+	}
+
+The possible simple data types are strings, integers or floats and boolean.
+
+.. tip:: The configuration and CLI syntax is Lua language, with which you may already be familiar with.
+         If not, you can read the `Learn Lua in 15 minutes`_ for a syntax overview. Spending just a few minutes
+         will allow you to break from static configuration, write more efficient configuration with iteration, and
+         leverage events and hooks. Lua is heavily used for scripting in applications ranging from embedded to game engines,
+         but in DNS world notably in `PowerDNS Recursor`_. Knot DNS Resolver does not simply use Lua modules, but it is
+         the heart of the daemon for everything from configuration, internal events and user interaction.
+
+Dynamic configuration
+^^^^^^^^^^^^^^^^^^^^^
+
+Knowing that the the configuration is a valid Lua script enables you to write dynamic rules, and also avoid
+additional configuration templating. One example is to differentiate between internal and external
+interfaces based on environment variable.
+
+.. code-block:: lua
+
+	if hostname() == 'hidden' then
+		net.listen(net.eth0)
+	else
+		net.listen(net.eth1.addr[1])
+	end
+
+Another example would show how it is possible to bind to all interfaces, using iteration.
+
+.. code-block:: lua
+
+	for name, addr_list in pairs(net.interfaces()) do
+		net.listen(addr_list)
+	end
+
+You can also use third-party packages (available for example through LuaRocks_) as on this example
+to download cache from parent, to avoid cold-cache start.
+
+.. code-block:: lua
+
+	local http = require('socket.http')
+	local ltn12 = require('ltn12')
+
+	if cache.count() == 0 then
+		-- download cache from parent
+		http.request { 
+			url = 'http://parent/cache.mdb',
+			sink = ltn12.sink.file(io.open('cache.mdb', 'w'))
+		}
+		-- reopen cache with 100M limit
+		cache.open('.', 100*MB)
+	end
+
+Events and services
+^^^^^^^^^^^^^^^^^^^
+
+The Lua supports a concept called closures, this is extremely useful for scripting actions upon various events.
+
+.. note:: Work in progress, come back later!
+
+* Timers and events
+* File watchers
+* Data I/O
+
+
+Configuration reference
+-----------------------
+
+This is a reference for variables and functions available to both configuration file and CLI.
+
+Environment
+^^^^^^^^^^^
+
+.. envvar:: env (table)
+
+   Return environment variable.
+
+   .. code-block:: lua
+
+	env.USER -- equivalent to $USER in shell
+
+.. function:: hostname()
+
+   :return: Machine hostname.
+
+Network configuration
+^^^^^^^^^^^^^^^^^^^^^
+
+.. function:: net.listen(address, [port = 53])
+
+   :return: boolean
+
+   Listen on address, port is optional.
+
+.. function:: net.listen({address1, ...}, [port = 53])
+
+   :return: boolean
+
+   Listen on list of addresses.
+
+.. function:: net.listen(interface, [port = 53])
+
+   :return: boolean
+
+   Listen on all addresses belonging to an interface.
+
+   Example:
+
+   .. code-block:: lua
+
+	net.listen(net.eth0) -- listen on eth0
+
+.. function:: net.close(address, [port = 53])
+
+   :return: boolean
+
+   Close opened address/port pair, noop if not listening.
+
+.. function:: net.list()
+
+   :return: Table of bound interfaces.
+
+   Example output:
+
+   .. code-block:: lua
+
+	[127.0.0.1] => {
+	    [port] => 53
+	    [tcp] => true
+	    [udp] => true
+	}
+
+.. function:: net.interfaces()
+
+   :return: Table of available interfaces and their addresses.
+
+   Example output:
+
+   .. code-block:: lua
+
+	[lo0] => {
+	    [addr] => {
+	        [1] => ::1
+	        [2] => 127.0.0.1
+	    }
+	    [mac] => 00:00:00:00:00:00
+	}
+	[eth0] => {
+	    [addr] => {
+	        [1] => 192.168.0.1
+	    }
+	    [mac] => de:ad:be:ef:aa:bb
+	}
+
+   .. tip:: You can use ``net.<iface>`` as a shortcut for specific interface, e.g. ``net.eth0``
+
+Modules configuration
+^^^^^^^^^^^^^^^^^^^^^
+
+The daemon provides an interface for dynamic loading of :ref:`daemon modules <modules-implemented>`.
+
+.. tip:: Use syntactic sugar for module loading. Declaring a variable ``modules`` equals to loading a table of modules.
+
+         .. code-block:: lua
+
+		modules = { hints = {file = '/etc/hosts'} }
+
+         Equals to:
+
+         .. code-block:: lua
+
+		modules.load('cachectl')
+		cachectl.config({file = '/etc/hosts'})
+
+
+.. function:: modules.list()
+
+   :return: List of loaded modules.
+
+.. function:: modules.load(name)
+
+   :param string name: Module name, e.g. "hints"
+   :return: boolean
+
+   Load a module by name.
+
+.. function:: modules.unload(name)
+
+   :param string name: Module name
+   :return: boolean
+
+   Unload a module by name.
+
+Cache configuration
+^^^^^^^^^^^^^^^^^^^
+
+The cache in Knot DNS Resolver is persistent with LMDB backend, this means that the daemon doesn't lose
+the cached data on restart or crash to avoid cold-starts. Interestingly the cache may be reused between cache
+daemons or manipulated from other processes, making for example synchronisation between load-balanced recursors possible.
+
+.. function:: cache.open(max_size)
+
+   :param number max_size: Maximum cache size in bytes.
+   :return: boolean
+
+   Open cache with size limit. The cache will be reopened if already open.
+   Note that the max_size cannot be lowered, only increased due to how cache is implemented.
+
+   .. tip:: Use ``kB, MB, GB`` constants as a multiplier, e.g. ``100*MB``.
+
+.. function:: cache.count()
+
+   :return: Number of entries in the cache.
+
+.. function:: cache.close()
+
+   :return: boolean
+
+   Close the cache.
+
+.. _`JSON-encoded`: http://json.org/example
+.. _`Learn Lua in 15 minutes`: http://tylerneylon.com/a/learn-lua/
+.. _`PowerDNS Recursor`: https://doc.powerdns.com/md/recursor/scripting/
+.. _LuaRocks: https://rocks.moonscript.org/
+.. _libuv: https://github.com/libuv/libuv
+.. _Lua: http://www.lua.org/about.html
+.. _LuaJIT: http://luajit.org/luajit.html
\ No newline at end of file
diff --git a/doc/config.rst b/doc/config.rst
deleted file mode 100644
index 870950b2265b736d6f5fb641a86975a163c1e54f..0000000000000000000000000000000000000000
--- a/doc/config.rst
+++ /dev/null
@@ -1,51 +0,0 @@
-Daemon configuration
---------------------
-
-The Knot DNS Resolver daemon has no traditional concept of static configuration.
-In it's simplest form it requires just a working directory in which it can set up persistent files like
-cache and the process state.
-
-.. code-block:: sh
-
-	$ kresolved /var/run/kresolved
-
-And you're good to go!
-
-Introduction
-~~~~~~~~~~~~
-
-There are several choices on how you can configure the daemon, a RPC interface a CLI or a configuration file,
-but fortunately all share a common syntax and are transparent to each other, e.g. if you change a knob, you're going to
-see it projected to other interfaces as well.
-
-.. note:: Expect this page to change a lot, as it's still just a proof of concept implementation.
-
-Configuration 101
-~~~~~~~~~~~~~~~~~
-
-If there is a `config` file in the daemon working directory, it's going to get loaded automatically, if there isn't one
-the daemon is going to start with sane defaults and listening on `localhost`. The syntax for options is like follows: ``group.option = value``
-or ``group.action(parameters)``. You can also comment using a ``--`` prefix.
-
-A simple example would be to increase the cache size.
-
-.. code-block:: lua
-
-	-- increase the cache to 100MB
-	cache.open(".", 100*1024*1024)
-
-Dynamic configuration
-~~~~~~~~~~~~~~~~~~~~~
-
-Packages and services
-~~~~~~~~~~~~~~~~~~~~~
-
-The Lua supports a concept called closures, this is extremely useful for scripting actions upon various events.
-
-.. note:: TODO, come back later!
-
-* Timers and events
-* File watchers
-* Serialization
-* Data I/O
-
diff --git a/doc/daemon.rst b/doc/daemon.rst
index 1cd4004a449c6f6ea86d52d0f40a1034a2c4ebab..dbeecabfa9755cac060255c1d7fd96987becad72 100644
--- a/doc/daemon.rst
+++ b/doc/daemon.rst
@@ -1,36 +1 @@
-Knot DNS Resolver daemon 
-========================
-
-Requirements
-------------
-
-* libuv_ 1.0+ (a multi-platform support library with a focus on asynchronous I/O)
-
-Starting the daemon
--------------------
-
-There is a separate resolver library in the `lib` directory, and a minimalistic daemon in
-the `daemon` directory. The daemon accepts a few CLI parameters, and there's no support for configuration
-right now.
-
-.. code-block:: bash
-
-	$ ./daemon/kresolved -h
-	$ ./daemon/kresolved -a 127.0.0.1#53
-
-.. _libuv: https://github.com/libuv/libuv
-
-Interacting with the daemon
----------------------------
-
-The daemon features a CLI interface if launched interactively, type ``help`` to see the list of available commands.
-You can load modules this way and use their properties to get information about statistics and such.
-
-.. code-block:: bash
-
-	$ kresolved /var/run/knot-resolver
-	...
-	[system] started in interactive mode, type 'help()'
-	> modules.load('cachectl')
-	> return cachectl.size()
-	{ "size": 53 }
\ No newline at end of file
+.. include:: ../daemon/README.rst
\ No newline at end of file