diff --git a/doc/modules.rst b/doc/modules.rst
index 428e09530bf086240a421985f88bb2e87707ec5c..21a2d9e51685b31385717cf8a0f15f1c4809d9c6 100644
--- a/doc/modules.rst
+++ b/doc/modules.rst
@@ -1,7 +1,13 @@
 .. include:: ../modules/README.rst
 
+.. _modules-implemented:
+
 Implemented modules
--------------------
+===================
+
+.. contents::
+   :depth: 1
+   :local:
 
 .. include:: ../modules/hints/README.rst
-.. include:: ../modules/cachectl/README.rst
+.. include:: ../modules/cachectl/README.rst
\ No newline at end of file
diff --git a/modules/README.rst b/modules/README.rst
index ec9f27ef64dee90aa8e12ce515eb5b694e052c82..92aba06a2f5d75fd36c372bb489055548a4c6e24 100644
--- a/modules/README.rst
+++ b/modules/README.rst
@@ -2,90 +2,27 @@
 Knot DNS Resolver extensions
 ****************************
 
-The resolver :ref:`library <lib_index>` leverages the `processing API`_ from the libknot to separate packet processing code
-into layers. In order to keep the core library sane and coverable, there are only two built-in layers:
-the :c:func:`iterate_layer`, and the :c:func:`itercache_layer`. The resolver context however can
-load shared libraries on runtime, which allows us to build and register external modules as well.
+Writing extensions
+==================
+
+.. contents::
+   :depth: 2
+   :local:
 
 Supported languages
 -------------------
 
 Currently modules written in C are supported.
-There is also a rudimentary support for writing modules in Go |---| ⑴ the library has no native Go bindings, library is accessible using CGO_, ⑵ gc doesn't support building shared libraries, GCCGO_ is required, ⑶ no coroutines and no garbage collecting thread, as the Go code is called from C threads.
-
-There is a plan for Lua scriptables, but it's not implemented yet.
-
-Available services
-------------------
-
-*Note* |---| This is only crash-course in the library internals, see the resolver :ref:`library <lib_index>` documentation for the complete overview of the services.
-
-The library offers following services:
-
-- :ref:`Cache <lib_cache>` - MVCC cache interface for retrieving/storing resource records.
-- :ref:`Resolution plan <lib_rplan>` - Query resolution plan, a list of partial queries (with hierarchy) sent in order to satisfy original query. This contains information about the queries, nameserver choice, timing information, answer and its class.
-- :ref:`Nameservers <lib_nameservers>` - Reputation database of nameservers, this serves as an aid for nameserver choice.
-
-If you're going to publish a layer in your module, it's going to be called by the query resolution driver for each query,
-so you're going to work with :ref:`struct kr_layer_param <lib_api_rplan>` as your per-query context. This structure contains pointers to
-resolution context, resolution plan and also the final answer. You're likely to retrieve currently solved query from the query plan:
-
-.. code-block:: c
-
-	int consume(knot_layer_t *ctx, knot_pkt_t *pkt)
-	{
-		struct kr_layer_param *param = ctx->data;
-		struct kr_query *query = kr_rplan_current(param->rplan);
-	}
+There is also a rudimentary support for writing modules in Go |---| |(1)| the library has no native Go bindings, library is accessible using CGO_, |(2)| gc doesn't support building shared libraries, GCCGO_ is required, |(3)| no coroutines and no garbage collecting thread, as the Go code is called from C threads.
 
-This is only passive processing of the incoming answer. If you want to change the course of resolution, say satisfy a query from a local cache before the library issues a query to the nameserver, you can use states (see the `Static hints`_ for example).
-
-.. code-block:: c
-
-	int produce(knot_layer_t *ctx, knot_pkt_t *pkt)
-	{
-		struct kr_layer_param *param = ctx->data;
-		struct kr_query *cur = kr_rplan_current(param->rplan);
-		
-		/* Query can be satisfied locally. */
-		if (can_satisfy(cur)) {
-			/* This flag makes the resolver move the query
-			 * to the "resolved" list. */
-			query->resolved = true;
-			return KNOT_STATE_DONE;
-		}
-
-		/* Pass-through. */
-		return ctx->state;
-	}
-
-It is possible to not only act during the query resolution, but also to view the complete resolution plan afterwards.
-This is useful for analysis-type tasks, or *"on-resolution"* hooks.
-
-.. code-block:: c
-
-	int finish(knot_layer_t *ctx)
-	{
-		struct kr_layer_param *param = ctx->data;
-		struct kr_rplan *rplan = param->rplan;
-
-		/* Print the query sequence with start time. */
-		char qname_str[KNOT_DNAME_MAXLEN];
-		struct kr_query *qry = NULL
-		WALK_LIST(qry, rplan->resolved) {
-			knot_dname_to_str(qname_str, qry->sname, sizeof(qname_str));
-			printf("%s at %u\n", qname_str, qry->timestamp);
-		}
-
-		return ctx->state;
-	}
+.. note:: There is a plan for Lua scriptables, but it's not implemented yet.
 
 The anatomy of an extension
 ---------------------------
 
 A module is a shared library defining specific functions, here's an overview of the functions.
 
-*Note* |---| the :ref:`Modules <lib_modules>` header documents the module loading and API.
+*Note* |---| the :ref:`Modules <lib_api_modules>` header documents the module loading and API.
 
 .. csv-table::
    :header: "C", "Go", "Params", "Comment"
@@ -94,7 +31,7 @@ A module is a shared library defining specific functions, here's an overview of
    "``X_init()``",    "``Init()``",   "``module``",      "Constructor"
    "``X_deinit()``",  "``Deinit()``", "``module, key``", "Destructor"
    "``X_config()``",  "``Config()``", "``module``",      "Configuration"
-   "``X_layer()``",   "``Layer()``",  "",                "Module layer"
+   "``X_layer()``",   "``Layer()``",  "",                ":ref:`Module layer <lib-layers>`"
    "``X_props()``",   "``Props()``",  "",                "NULL-terminated list of properties"
 
 .. [#] Mandatory symbol.
@@ -102,12 +39,10 @@ A module is a shared library defining specific functions, here's an overview of
 The ``X_`` corresponds to the module name, if the module name is ``hints``, then the prefix for constructor would be ``hints_init()``.
 This doesn't apply for Go, as it for now always implements `main` and requires capitalized first letter in order to export its symbol.
 
-How does the module get loaded
-~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
-
-The resolution context :c:type:`struct kr_context` holds loaded modules for current context. A module can be registered with :c:func:`kr_context_register`, which triggers module constructor *immediately* after the load. Module destructor is automatically called when the resolution context closes.
-
-If the module exports a layer implementation, it is automatically discovered by :c:func:`kr_resolver` on resolution init and plugged in. The order in which the modules are registered corresponds to the call order of layers.
+.. note::
+   The resolution context :c:type:`struct kr_context` holds loaded modules for current context. A module can be registered with :c:func:`kr_context_register`, which triggers module constructor *immediately* after the load. Module destructor is automatically called when the resolution context closes.
+   
+   If the module exports a layer implementation, it is automatically discovered by :c:func:`kr_resolver` on resolution init and plugged in. The order in which the modules are registered corresponds to the call order of layers.
 
 Writing a module in C
 ---------------------
@@ -123,7 +58,7 @@ As almost all the functions are optional, the minimal module looks like this:
 
 Let's define an observer thread for the module as well. It's going to be stub for the sake of brevity,
 but you can for example create a condition, and notify the thread from query processing by declaring
-module layer (see the `Available services`_).
+module layer (see the :ref:`Writing layers <lib-layers>`).
 
 .. code-block:: c
 
@@ -244,6 +179,8 @@ Configuring modules
 
 There is a callback ``X_config()`` but it's NOOP for now, as the configuration is not yet implemented.
 
+.. _mod-properties:
+
 Exposing module properties
 --------------------------
 
@@ -286,7 +223,7 @@ Here's an example how a module can expose its property:
 	{
 		static struct kr_prop prop_list[] = {
 			/* Callback,   Name,   Description */
-			{&get_size, "size", "Return number of records."},
+			{&get_size, "get_size", "Return number of records."},
 			{NULL, NULL, NULL}
 		};
 		return prop_list;
@@ -302,14 +239,16 @@ Once you load the module, you can call the module property from the interactive
 	...
 	[system] started in interactive mode, type 'help()'
 	> modules.load('cached')
-	> return cached.size()
+	> cached.get_size()
 	{ "size": 53 }
 
 *Note* |---| this relies on function pointers, so the same ``static inline`` trick as for the ``Layer()`` is required for C/Go.
 
-.. _`processing API`: https://gitlab.labs.nic.cz/labs/knot/tree/master/src/libknot/processing
 .. _`not present in Go`: http://blog.golang.org/gos-declaration-syntax
 .. _CGO: http://golang.org/cmd/cgo/
 .. _GCCGO: https://golang.org/doc/install/gccgo
 
 .. |---| unicode:: U+02014 .. em dash
+.. |(1)| unicode:: U+2474 .. (1)
+.. |(2)| unicode:: U+2475 .. (2)
+.. |(3)| unicode:: U+2476 .. (3)
diff --git a/modules/hints/README.rst b/modules/hints/README.rst
index 6c6e9b966d9ffbfd7f650b0577e4c37427170756..627f79954250b2fe20011b0914906ba842cd288e 100644
--- a/modules/hints/README.rst
+++ b/modules/hints/README.rst
@@ -1,4 +1,6 @@
+.. _mod-hints:
+
 Static hints
-~~~~~~~~~~~~
+------------
 
 This is a module providing static hints from ``/etc/hosts``, document me.
\ No newline at end of file