docs: add json configuration details

Signed-off-by: Patrick Venture <venture@google.com>
Change-Id: I695e732b2c32bd96d6d68500a8af493a840dc79c
diff --git a/README.md b/README.md
index 5324b1c..1b4c244 100644
--- a/README.md
+++ b/README.md
@@ -192,6 +192,10 @@
 `VERIFY_BIOS_TARGET`          | `phosphor-ipmi-flash-bios-verify.target`  | The systemd target started for verification.
 `UPDATE_BIOS_TARGET`          | `phosphor-ipmi-flash-bios-update.target`  | The systemd target started for updating the BIOS.
 
+## JSON Configuration
+
+Read the [details](bmc_json_config.md) of the json configuration.  The json configurations are used to configure the BMC's flash handler behaviors.
+
 ## Flash State Machine Details
 
 [This document](ipmi_flash.md) describes the details of the state machine
diff --git a/bmc_json_config.md b/bmc_json_config.md
new file mode 100644
index 0000000..3480bbb
--- /dev/null
+++ b/bmc_json_config.md
@@ -0,0 +1,280 @@
+# BMC Handler Configuration
+
+The BMC BLOB handler for the flash objects is configured at run-time via a json
+file or multiple json files. The json fields detail every aspect of how the
+handler should behave for a given blob id.
+
+There are three json configurations available by default:
+
+*   config-bios.json
+*   config-static-bmc-reboot.json
+*   config-static-bmc.json
+
+Let's break them down, what they contain and what it means, and then how to
+write your own. It's helpful to start here.
+
+## config-bios.json
+
+This file is generated from configuration and therefore some values can be
+replaced, here we're using the defaults.
+
+```
+[{
+    "blob": "/flash/bios",
+    "handler": {
+        "type": "file",
+        "path": "/tmp/bios-image"
+    },
+    "actions": {
+        "preparation": {
+            "type": "systemd",
+            "unit": "phosphor-ipmi-flash-bios-prepare.target"
+        },
+        "verification": {
+            "type": "fileSystemdVerify",
+            "unit": "phosphor-ipmi-flash-bios-verify.target",
+            "path": "/tmp/bios.verify"
+        },
+        "update": {
+            "type": "systemd",
+            "unit": "phosphor-ipmi-flash-bios-update.target"
+        }
+    }
+}]
+```
+
+Each json file is expected to be an array of flash handler configurations. You
+can define more than one handler inside a json file, or you can have one per
+file. It will make no difference. The file though, must contain an array.
+
+Beyond this, the flash handler configuration is a dictionary or object with
+three fields.
+
+*   `blob`
+*   `handler`
+*   `actions`
+
+### `blob`
+
+The `blob` field expects to hold a string. This string is the blob id that'll be
+presented to the host and registered with the blob manager. It must be unique
+among all blobs on a system and must start with `/flash/`. The uniqueness is not
+presently verified at any stage. If there is another blob id that matches, the
+first one loaded will be the first one hit during any calls from the host. It's
+generally unlikely that there will be a name collision.
+
+### `handler`
+
+The `handler` field expects to hold a dictionary or object with at least one
+field.
+
+*   `type`
+
+The `type` field expects a string and will control what other parameters are
+required. The type here refers to the handler for the incoming data associated
+with the flash image contents. The hash data is always done as a file, but the
+image data is configurable via the json.
+
+In the above configuration the `type` is set to `file`. The `file` type handler
+requires one parameter: `path`. The `path` here is the full file system path to
+where to write the bytes received into a file. In this case specifically the
+byte received will be written to `/tmp/bios-image`.
+
+### `actions`
+
+Because `phosphor-ipmi-flash` is a framework for sending data from the host to
+the BMC for updating various firmware, it needs to be told what to do to verify
+the data and cause an update. It additionally provides a mechanism for preparing
+the BMC to receive the data. On a low memory BMC, the preparation step can be
+used to shut down unnecessary services, etc.
+
+The actions are split into three fields:
+
+*   `preparation`
+*   `verification`
+*   `update`
+
+The `preparation`, `verification`, and `update` fields expect a `type` field,
+similarly to the `handler` field. This dictates what other parameters may be
+required.
+
+In this configuration the `preparation` type is `systemd`. The `systemd` type
+expects to receive a `unit`. The `unit` value is expected to be a string that is
+the name of the systemd unit to start. In this case, it'll start
+`phosphor-ipmi-flash-bios-prepare.target`. This can be a single service name, or
+a target.
+
+In this configuration the `verification` type is `fileSystemdVerify`. This is
+similar to the `systemd` type except it also expects a `path`. The `path` is
+used to read back the status of whatever was triggered via the `unit` field. Was
+the verification successful? Is the verification still in progress? This
+information is read back from the file specified by the `path` field.
+
+In this configuration the `update` type is `systemd`. This is the same object as
+with the `preparation` action. In this case, though the status cannot be read
+back with something useful. Typically, the `fileSystemdVerify` type should be
+used as it provides information on success.
+
+## config-static-bmc-reboot.json
+
+This file is generated from configuration and therefore some values can be
+replaced, here we're using the defaults.
+
+```
+[{
+    "blob": "/flash/image",
+    "handler": {
+        "type": "file",
+        "path": "/run/initramfs/bmc-image"
+    },
+    "actions": {
+        "preparation": {
+            "type": "systemd",
+            "unit": "phosphor-ipmi-flash-bmc-prepare.target"
+        },
+        "verification": {
+            "type": "fileSystemdVerify",
+            "unit": "phosphor-ipmi-flash-bmc-verify.target",
+            "path": "/tmp/bmc.verify"
+        },
+        "update": {
+            "type": "reboot"
+        }
+    }
+}]
+```
+
+Given the bios configuration, we can skip the parts of this configuration that
+are already explained. The new action type is `reboot`. This is a special action
+type that has no parameters. It will simply trigger a reboot via systemd.
+
+## config-static-bmc.json
+
+This file is generated from configuration and therefore some values can be
+replaced, here we're using the defaults.
+
+```
+[{
+    "blob": "/flash/image",
+    "handler": {
+        "type": "file",
+        "path": "/run/initramfs/bmc-image"
+    },
+    "actions": {
+        "preparation": {
+            "type": "systemd",
+            "unit": "phosphor-ipmi-flash-bmc-prepare.target"
+        },
+        "verification": {
+            "type": "fileSystemdVerify",
+            "unit": "phosphor-ipmi-flash-bmc-verify.target",
+            "path": "/tmp/bmc.verify"
+        },
+        "update": {
+            "type": "systemd",
+            "unit": "phosphor-ipmi-flash-bmc-update.target"
+        }
+    }
+}]
+```
+
+This configuration is of no significance in its difference.
+
+## Writing your own json Configuration
+
+If you wish to write your own json configuration, you simply create a legal json
+file containing an array of at least one entry. This array will define any
+entries you wish. You can provide multiple BMC update json configurations if you
+wish. There may be parameters you desire to pass to an update process, for
+instance, clearing the whitelisted files. In this case, it's trivial to create
+two targets. The first target performs a normal update, the second starts a
+service that performs the desired other update. You can leverage the
+default-provided configuration files, and also add your own to add additional
+support if desired.
+
+An entry, regardless, must contain a `blob`, a `handler`, and `actions`.
+
+### Handler Types
+
+A handler determines how the bytes from the host are stored.
+
+#### `file`
+
+The `file` handler type writes the bytes to a file path specified by the
+required parameter `path`.
+
+*   `path` - full file system path to where to write bytes.
+
+### Action Types
+
+Action types are used to define what to do for a specific requested action, such
+as "verify the blob contents."
+
+#### `systemd`
+
+The `systemd` type should be used when you wish to start a systemd service or
+target but only care about the status of whether or not you were able to start
+it, not whether it did what you ultimately wanted.
+
+*   `unit` - required - string - the systemd unit to start.
+*   `mode` - optional - string - default: replace - the mode for starting the
+    service.
+
+#### `fileSystemdVerify`
+
+Because one may care about the result of their actions, the `fileSystemdVerify`
+action type exists. It will start the service and when asked for a status, it'll
+read the contents of a file. Therefore, whatever is performing the action will
+want to update that file.
+
+*   `path` - required - string - the full file system path to where one finds
+    the status.
+*   `unit` - required - string - the systemd unit to start
+*   `mode` - optional - string - default: replace - the mode for starting the
+    service.
+
+#### `reboot`
+
+The `reboot` type causes a reboot via systemd. If this happens quickly enough,
+you may not receive a response over IPMI that your command succeeded.
+
+### Installing your own json configuration
+
+To install your own configuration(s) you can add your own bbappend file that
+appends to the install task. The json files are installed in
+`${D}${datadir}/phosphor-ipmi-flash/`
+
+## Adding your own handler type
+
+Firstly, welcome! The maintainers of this repository appreciate your
+contribution. A handler is just an implementation of the
+`ImageHandlerInterface`.
+
+Your handler must implement:
+
+*   `bool open(const std::string& path)`
+*   `void close()`
+*   `bool write(std::uint32_t offset, const std::vector<std::uint8_t>& data)`
+*   `int getSize()`
+
+The handler is meant to receive the bytes, and write the bytes.
+
+Once the object is defined you can specify a type and parameters here and in the
+`buildjson` module.
+
+## Adding your own action type
+
+Firstly, welcome! The maintainers of this repository appreciate your
+contribution. An action is just an implementation of the
+`TriggerableActionInterface`.
+
+Your action must implement:
+
+*   `bool trigger()`
+*   `void abort()`
+*   `ActionStatus status()`
+
+The abort method is not a guarantee.
+
+Once the object is defined you can specify a type and parameters here and in the
+`buildjson` module.