* AT91's Analog to Digital Converter (ADC)
 
 Required properties:
-  - compatible: Should be "atmel,at91sam9260-adc"
+  - compatible: Should be "atmel,<chip>-adc"
+    <chip> can be "at91sam9260", "at91sam9g45" or "at91sam9x5"
   - reg: Should contain ADC registers location and length
   - interrupts: Should contain the IRQ line for the ADC
-  - atmel,adc-channel-base: Offset of the first channel data register
   - atmel,adc-channels-used: Bitmask of the channels muxed and enable for this
     device
-  - atmel,adc-drdy-mask: Mask of the DRDY interruption in the ADC
   - atmel,adc-num-channels: Number of channels available in the ADC
   - atmel,adc-startup-time: Startup Time of the ADC in microseconds as
     defined in the datasheet
-  - atmel,adc-status-register: Offset of the Interrupt Status Register
-  - atmel,adc-trigger-register: Offset of the Trigger Register
   - atmel,adc-vref: Reference voltage in millivolts for the conversions
   - atmel,adc-res: List of resolution in bits supported by the ADC. List size
                   must be two at least.
 
 #define at91_adc_writel(st, reg, val) \
        (writel_relaxed(val, st->reg_base + reg))
 
+struct at91_adc_caps {
+       struct at91_adc_reg_desc registers;
+};
+
 struct at91_adc_state {
        struct clk              *adc_clk;
        u16                     *buffer;
        u32                     res;            /* resolution used for convertions */
        bool                    low_res;        /* the resolution corresponds to the lowest one */
        wait_queue_head_t       wq_data_avail;
+       struct at91_adc_caps    *caps;
 };
 
 static irqreturn_t at91_adc_trigger_handler(int irq, void *p)
        return ret;
 }
 
+static const struct of_device_id at91_adc_dt_ids[];
+
 static int at91_adc_probe_dt(struct at91_adc_state *st,
                             struct platform_device *pdev)
 {
        if (!node)
                return -EINVAL;
 
+       st->caps = (struct at91_adc_caps *)
+               of_match_device(at91_adc_dt_ids, &pdev->dev)->data;
+
        st->use_external = of_property_read_bool(node, "atmel,adc-use-external-triggers");
 
        if (of_property_read_u32(node, "atmel,adc-channels-used", &prop)) {
        if (ret)
                goto error_ret;
 
-       st->registers = devm_kzalloc(&idev->dev,
-                                    sizeof(struct at91_adc_reg_desc),
-                                    GFP_KERNEL);
-       if (!st->registers) {
-               dev_err(&idev->dev, "Could not allocate register memory.\n");
-               ret = -ENOMEM;
-               goto error_ret;
-       }
-
-       if (of_property_read_u32(node, "atmel,adc-channel-base", &prop)) {
-               dev_err(&idev->dev, "Missing adc-channel-base property in the DT.\n");
-               ret = -EINVAL;
-               goto error_ret;
-       }
-       st->registers->channel_base = prop;
-
-       if (of_property_read_u32(node, "atmel,adc-drdy-mask", &prop)) {
-               dev_err(&idev->dev, "Missing adc-drdy-mask property in the DT.\n");
-               ret = -EINVAL;
-               goto error_ret;
-       }
-       st->registers->drdy_mask = prop;
-
-       if (of_property_read_u32(node, "atmel,adc-status-register", &prop)) {
-               dev_err(&idev->dev, "Missing adc-status-register property in the DT.\n");
-               ret = -EINVAL;
-               goto error_ret;
-       }
-       st->registers->status_register = prop;
-
-       if (of_property_read_u32(node, "atmel,adc-trigger-register", &prop)) {
-               dev_err(&idev->dev, "Missing adc-trigger-register property in the DT.\n");
-               ret = -EINVAL;
-               goto error_ret;
-       }
-       st->registers->trigger_register = prop;
-
+       st->registers = &st->caps->registers;
        st->trigger_number = of_get_child_count(node);
        st->trigger_list = devm_kzalloc(&idev->dev, st->trigger_number *
                                        sizeof(struct at91_adc_trigger),
 }
 
 #ifdef CONFIG_OF
+static struct at91_adc_caps at91sam9260_caps = {
+       .registers = {
+               .channel_base = AT91_ADC_CHR(0),
+               .drdy_mask = AT91_ADC_DRDY,
+               .status_register = AT91_ADC_SR,
+               .trigger_register = AT91_ADC_TRGR_9260,
+       },
+};
+
+static struct at91_adc_caps at91sam9g45_caps = {
+       .registers = {
+               .channel_base = AT91_ADC_CHR(0),
+               .drdy_mask = AT91_ADC_DRDY,
+               .status_register = AT91_ADC_SR,
+               .trigger_register = AT91_ADC_TRGR_9G45,
+       },
+};
+
+static struct at91_adc_caps at91sam9x5_caps = {
+       .registers = {
+               .channel_base = AT91_ADC_CDR0_9X5,
+               .drdy_mask = AT91_ADC_SR_DRDY_9X5,
+               .status_register = AT91_ADC_SR_9X5,
+               .trigger_register = AT91_ADC_TRGR_9X5,
+       },
+};
+
 static const struct of_device_id at91_adc_dt_ids[] = {
-       { .compatible = "atmel,at91sam9260-adc" },
+       { .compatible = "atmel,at91sam9260-adc", .data = &at91sam9260_caps },
+       { .compatible = "atmel,at91sam9g45-adc", .data = &at91sam9g45_caps },
+       { .compatible = "atmel,at91sam9x5-adc", .data = &at91sam9x5_caps },
        {},
 };
 MODULE_DEVICE_TABLE(of, at91_adc_dt_ids);