diff --git a/docs/framework/routing.md b/docs/framework/routing.md
new file mode 100644
index 0000000000000000000000000000000000000000..c8a6389d953a54f2e305987950b5afbe703adbf5
--- /dev/null
+++ b/docs/framework/routing.md
@@ -0,0 +1,209 @@
+# Routing
+
+CiviCRM's routing system is built based on XML files. These XML files define what class gets loaded when a specific path is arrived at, what permissions are required to access that route.
+
+The standard menu XML files can be found in `CRM/Core/xml/Menu/`. Each route is defined as an "Item" Within the menu. In extensions you should add your menu to `<extension folder>/xml/Menu/<extensionName>.xml`
+
+!!! note
+    For historical reasons, the routing files live in a `Menu` folder, but the contents of these files do *not* affect the navigation menu at the top of the screen. 
+    
+    Extension authors can add new menu entires by using [hook_civicrm_navigationMenu](/hooks/hook_civicrm_navigationMenu.md).
+    
+## Example
+
+```xml
+<menu>
+  <item>
+    <path>civicrm/payment/form</path>
+    <access_callback>1</access_callback>
+    <page_callback>CRM_Financial_Form_Payment</page_callback>
+    <is_public>true</is_public>
+    <weight>0</weight>
+  </item>
+  <item>
+    <path>civicrm/payment/edit</path>
+    <page_callback>CRM_Financial_Form_PaymentEdit</page_callback>
+    <access_arguments>access CiviContribute</access_arguments>
+    <component>CiviContribute</component>
+  </item>
+</menu>
+```
+
+## Elements
+
+The XML will contain a structure made up of the following elements.
+
+!!! tip
+    The [`<menu>`](#menu) element must be the root element of the document.
+    
+### `<access_callback>` {:#access_callback}
+
+* Containing element: [`<item>`](#item)
+* Description: Function to be used to check access to the route
+* Example: `CRM_Core_Permission::checkMenu`
+* Contains: Text
+* Notes:
+    * If you wish for this route to be public you can set it to be 1.
+
+### `<access_arguments>` {:#access_arguments}
+
+* Containing element: [`<item>`](#item)
+* Description: Arguments to be passed to the permission checking function
+* Example: `access CiviCRM;administer CiviCRM`
+* Contains: Text
+* Notes:
+    * If you want the permissions to be an "or" situation i.e. User needs either access CiviCRM or administer CiviCRM put a `;` between the permissions. If you want it so that users need multiple permissions put a `,` between
+
+### `<adminGroup>` {:#adminGroup}
+
+* Containing element: [`<item>`](#item)
+* Description: Is this menu part of a Administration group and if so which one
+* Example: `Manage`
+* Contains: Text
+
+### `<comment>` {:#comment}
+
+* Containing element: [`<item>`](#item)
+* Description: ???
+* Contains: Text
+
+### `<component>` {:#component}
+
+* Containing element: [`<item>`](#item)
+* Description: ???
+* Example: `CiviContribute`
+* Contains: Text
+
+### `<desc>` {:#desc}
+
+* Containing element: [`<item>`](#item)
+* Description: What is the description of this route
+* Example: `This sets up specific eway settings`
+* Contains: Text
+
+### `<icon>` {:#icon}
+
+* Containing element: [`<item>`](#item)
+* Description: What icon should display next to the title in the menu
+* Example: `admin/small/duplicate_matching.png`
+* Contains: Text
+
+### `<item>` {:#item}
+
+* Containing element: [`<menu>`](#menu)
+* Contains: Elements
+
+Elements acceptable within `<item>`
+
+| Element | Acceptable instances | 
+| -- | -- |
+| [`<access_callback>`](#access_callback) | 0 or 1 |
+| [`<access_arguments>`](#access_arguments) | 0 or 1 |
+| [`<adminGroup>`](#adminGroup) | 0 or 1 |
+| [`<desc>`](#desc) | 0 or 1 |
+| [`<icon>`](#icon) | 0 or 1 |
+| [`<is_public>`](#is_public) | 0 or 1 |
+| [`<is_ssl>`](#is_ssl) | 0 or 1 |
+| [`<comment>`](#comment) | 0 or 1 |
+| [`<component>`](#component) | 0 or 1 |
+| [`<path>`](#path) | 1 |
+| [`<page_arguments>`](#page_arguments) | 0 or 1 |
+| [`<page_callback>`](#page_callback) | 0 or 1 |
+| [`<page_type>`](#page_type) | 0 or 1 |
+| [`<path_arguments>`](#path_arguments) | 0 or 1 |
+| [`<return_url>`](#return_url) | 0 or 1 |
+| [`<skipBreadCrumb>`](#skipBreadCrumb) | 0 or 1 |
+| [`<title>`](#title) | 1 |
+| [`<weight>`](#weight) | 0 or 1 |
+
+### `<is_public>` {:#is_public}
+
+* Containing element: [`<item>`](#item)
+* Description: ???
+* Contains: `true` or `false`
+
+### `<is_ssl>` {:#is_ssl}
+
+* Containing element: [`<item>`](#item)
+* Description: ???
+* Contains: `true` or `false`
+
+### `<menu>` {:#menu}
+
+* Containing element: none (this is the root element)
+* Contains: Elements
+
+Elements acceptable within `<menu>`
+
+| Element | Acceptable instances | 
+| -- | -- |
+| [`<item>`](#item) | 1+ |
+
+### `<path>` {:#path}
+
+* Containing element: [`<item>`](#item)
+* Description: The path is the url route that this menu item is for
+* Example: `civicrm/admin/eway/settings`
+* Contains: Text
+
+!!! Caution "Caution: Wild card sub-paths"
+    One path can match all sub-paths.  For example, `<path>civicrm/admin</path>` can match `http://example.org/civicrm/admin/f/o/o/b/a/r`.  However, one should avoid designs which rely on this because it's imprecise and it can be difficult to integrate with some frontends.
+
+
+### `<page_arguments>` {:#page_arguments}
+
+* Containing element: [`<item>`](#item)
+* Description: Arguments passed to the function generating the content of the route
+* Example: `addSequence=1`
+* Contains: Text
+
+### `<page_callback>` {:#page_callback}
+
+* Containing element: [`<item>`](#item)
+* Description: Function called to generate the content of the route
+* Example: `CRM_Contact_Page_Dashboard`
+* Contains: Text
+
+### `<page_type>` {:#page_type}
+
+* Containing element: [`<item>`](#item)
+* Description: What type of a page is this
+* Example: `1 or contribute`
+* Contains: Text
+* Notes:
+    * If this is not set the default is 0
+
+### `<path_arguments>` {:#path_arguments}
+
+* Containing element: [`<item>`](#item)
+* Description:  These are arguments to be added to the url when it is loaded. These are generally useful when redirecting people after filling in the page or form
+* Example `ct=Household`
+* Contains: Text
+
+### `<return_url>` {:#return_url}
+
+* Containing element: [`<item>`](#item)
+* Description: ???
+* Contains: Text
+
+### `<skipBreadCrumb>` {:#skipBreadCrumb}
+
+* Containing element: [`<item>`](#item)
+* Description: Should this route not generate any breadcrumbs
+* Contains: `true` or `false`
+
+### `<title>` {:#title}
+
+* Containing element: [`<item>`](#item)
+* Description: Title of the route
+* Example: `Eway Settings`
+* Contains: Text
+
+### `<weight>` {:#weight}
+
+* Containing element: [`<item>`](#item)
+* Description: Set a weight on the route to change the order of it in the menu
+* Example: `105`
+* Contains: Integer
+* Notes:
+    * Items with heavier weights will appear lower in the menu
diff --git a/docs/hooks/hook_civicrm_xmlMenu.md b/docs/hooks/hook_civicrm_xmlMenu.md
index 23ff3312e3a746bf301872b4bbf0067633308e7e..f4a69c92eb8ff0a8811f2e9404dc7ac6361a79a4 100644
--- a/docs/hooks/hook_civicrm_xmlMenu.md
+++ b/docs/hooks/hook_civicrm_xmlMenu.md
@@ -55,34 +55,9 @@ function EXAMPLE_civicrm_xmlMenu(&$files) {
 }
 ```
 
-## XML: Common
-
-Several elements are supported in any route:
-
- * `<path>` (ex:`civicrm/ajax/my-page`): This specifies the URL of the page. On a system like Drupal (which supports "clean URLs"), the full page URL would look like `http://example.org/civicrm/ajax/my-page`.
- * `<page_callback>` (ex: `CRM_Example_Page_AJAX::runMyPage` or `CRM_Example_Page_MyStuff`): This specifies the page-controller, which may be any of the following:
-    * Static function (ex: `CRM_Example_Page_AJAX::runMyPage`)
-    * A subclass of `CRM_Core_Page` named `CRM_*_Page_*` (ex: `CRM_Example_Page_MyStuff`)
-    * A subclass of `CRM_Core_Form` named `CRM_*_Form_*` (ex: `CRM_Example_Form_MyStuff`)
-    * A subclass of `CRM_Core_Controller` named `CRM_*_Controller_*` (ex: `CRM_Example_Controller_MyStuff` )
- * `<access_arguments>` (ex: `administer CiviCRM`): A list of permissions required for this page.
-    * If you'd like to reference *new* permissions, be sure to declare them with [hook_civicrm_permission](/hooks/hook_civicrm_permission.md).
-    * To require *any one* permission from a list, use a semicolon (`;`). (ex: `edit foo;administer bar` requires `edit foo` **or** `administer bar`)
-    * To require *multiple* permissions, use a comma (`,`). (ex: `edit foo,administer bar` requires `edit foo` **and** `administer bar`)
-    * At time of writing, mixing `,`s and `;`s has not been tested.
- * `<title>` (ex: `Hello world`): Specifies the default value for the HTML `<TITLE>`. (This default be programmatically override on per-request basis.)
-
-!!! caution "Caution: Wildcard sub-paths"
-    One path can match all subpaths.  For example, `<path>civicrm/admin</path>` can match `http://example.org/civicrm/admin/f/o/o/b/a/r`.  However, one should avoid designs which rely on this because it's imprecise and it can be difficult to integrate with some frontends.
-
-## XML: Administration
-
-The administration screen (`civicrm/admin`) includes a generated list of fields. This content is determined by some additional elements:
-
- * `<desc>`
- * `<icon>`
- * `<adminGroup>`
- * `<weight>`
+## XML Structure
+
+See the [routing](/framework/routing.md) page for details on the XML schema.
 
 ## XML: IDS
 
diff --git a/mkdocs.yml b/mkdocs.yml
index 318acfb066a2e74d528e91247bf184d752568d7c..ea1cff0eb5e882b6dbf3ec73b1300e6f61e3c05f 100644
--- a/mkdocs.yml
+++ b/mkdocs.yml
@@ -224,6 +224,7 @@ pages:
      - QuickForm: framework/quickform/index.md
      - Entity Reference Field: framework/quickform/entityref.md
   - Region Reference: framework/region.md
+  - Routing: framework/routing.md
   - Resources Reference: framework/resources.md
   - Setting Reference: framework/setting.md
   - Template Reference: