#include <linux/reboot.h>
 #include <linux/regmap.h>
 
-struct syscon_reboot_context {
-       struct regmap *map;
+struct reboot_mode_bits {
        u32 offset;
-       u32 value;
        u32 mask;
+       u32 value;
+       bool valid;
+};
+
+struct reboot_data {
+       struct reboot_mode_bits mode_bits[REBOOT_SOFT + 1];
+       struct reboot_mode_bits catchall;
+};
+
+struct syscon_reboot_context {
+       struct regmap *map;
+
+       const struct reboot_data *rd; /* from of match data, if any */
+       struct reboot_mode_bits catchall; /* from DT */
+
        struct notifier_block restart_handler;
 };
 
        struct syscon_reboot_context *ctx =
                        container_of(this, struct syscon_reboot_context,
                                        restart_handler);
+       const struct reboot_mode_bits *mode_bits;
+
+       if (ctx->rd) {
+               if (mode < ARRAY_SIZE(ctx->rd->mode_bits) &&
+                   ctx->rd->mode_bits[mode].valid)
+                       mode_bits = &ctx->rd->mode_bits[mode];
+               else
+                       mode_bits = &ctx->rd->catchall;
+       } else {
+               mode_bits = &ctx->catchall;
+       }
 
        /* Issue the reboot */
-       regmap_update_bits(ctx->map, ctx->offset, ctx->mask, ctx->value);
+       regmap_update_bits(ctx->map, mode_bits->offset, mode_bits->mask,
+                          mode_bits->value);
 
        mdelay(1000);
 
 {
        struct syscon_reboot_context *ctx;
        struct device *dev = &pdev->dev;
-       int mask_err, value_err;
        int priority;
        int err;
 
        if (of_property_read_s32(pdev->dev.of_node, "priority", &priority))
                priority = 192;
 
-       if (of_property_read_u32(pdev->dev.of_node, "offset", &ctx->offset))
-               if (of_property_read_u32(pdev->dev.of_node, "reg", &ctx->offset))
-                       return -EINVAL;
+       ctx->rd = of_device_get_match_data(dev);
+       if (!ctx->rd) {
+               int mask_err, value_err;
 
-       value_err = of_property_read_u32(pdev->dev.of_node, "value", &ctx->value);
-       mask_err = of_property_read_u32(pdev->dev.of_node, "mask", &ctx->mask);
-       if (value_err && mask_err) {
-               dev_err(dev, "unable to read 'value' and 'mask'");
-               return -EINVAL;
-       }
+               if (of_property_read_u32(pdev->dev.of_node, "offset",
+                                        &ctx->catchall.offset) &&
+                   of_property_read_u32(pdev->dev.of_node, "reg",
+                                        &ctx->catchall.offset))
+                       return -EINVAL;
 
-       if (value_err) {
-               /* support old binding */
-               ctx->value = ctx->mask;
-               ctx->mask = 0xFFFFFFFF;
-       } else if (mask_err) {
-               /* support value without mask*/
-               ctx->mask = 0xFFFFFFFF;
+               value_err = of_property_read_u32(pdev->dev.of_node, "value",
+                                                &ctx->catchall.value);
+               mask_err = of_property_read_u32(pdev->dev.of_node, "mask",
+                                               &ctx->catchall.mask);
+               if (value_err && mask_err) {
+                       dev_err(dev, "unable to read 'value' and 'mask'");
+                       return -EINVAL;
+               }
+
+               if (value_err) {
+                       /* support old binding */
+                       ctx->catchall.value = ctx->catchall.mask;
+                       ctx->catchall.mask = 0xFFFFFFFF;
+               } else if (mask_err) {
+                       /* support value without mask */
+                       ctx->catchall.mask = 0xFFFFFFFF;
+               }
        }
 
        ctx->restart_handler.notifier_call = syscon_restart_handle;
        return err;
 }
 
+static const struct reboot_data gs101_reboot_data = {
+       .mode_bits = {
+               [REBOOT_WARM] = {
+                       .offset = 0x3a00, /* SYSTEM_CONFIGURATION */
+                       .mask = 0x00000002, /* SWRESET_SYSTEM */
+                       .value = 0x00000002,
+                       .valid = true,
+               },
+               [REBOOT_SOFT] = {
+                       .offset = 0x3a00, /* SYSTEM_CONFIGURATION */
+                       .mask = 0x00000002, /* SWRESET_SYSTEM */
+                       .value = 0x00000002,
+                       .valid = true,
+               },
+       },
+       .catchall = {
+               .offset = 0x3e9c, /* PAD_CTRL_PWR_HOLD */
+               .mask = 0x00000100,
+               .value = 0x00000000,
+       },
+};
+
 static const struct of_device_id syscon_reboot_of_match[] = {
+       { .compatible = "google,gs101-reboot", .data = &gs101_reboot_data  },
        { .compatible = "syscon-reboot" },
        {}
 };