xref: /openbmc/linux/drivers/clk/clk-max9485.c (revision 2612e3bbc0386368a850140a6c9b990cd496a5ec)
133f51046SDaniel Mack // SPDX-License-Identifier: GPL-2.0
233f51046SDaniel Mack 
333f51046SDaniel Mack #include <linux/module.h>
433f51046SDaniel Mack #include <linux/kernel.h>
533f51046SDaniel Mack #include <linux/clk.h>
633f51046SDaniel Mack #include <linux/clk-provider.h>
733f51046SDaniel Mack #include <linux/err.h>
833f51046SDaniel Mack #include <linux/errno.h>
933f51046SDaniel Mack #include <linux/gpio/consumer.h>
1033f51046SDaniel Mack #include <linux/i2c.h>
1133f51046SDaniel Mack #include <linux/regulator/consumer.h>
1233f51046SDaniel Mack 
1333f51046SDaniel Mack #include <dt-bindings/clock/maxim,max9485.h>
1433f51046SDaniel Mack 
1533f51046SDaniel Mack #define MAX9485_NUM_CLKS 4
1633f51046SDaniel Mack 
1733f51046SDaniel Mack /* This chip has only one register of 8 bit width. */
1833f51046SDaniel Mack 
1933f51046SDaniel Mack #define MAX9485_FS_12KHZ	(0 << 0)
2033f51046SDaniel Mack #define MAX9485_FS_32KHZ	(1 << 0)
2133f51046SDaniel Mack #define MAX9485_FS_44_1KHZ	(2 << 0)
2233f51046SDaniel Mack #define MAX9485_FS_48KHZ	(3 << 0)
2333f51046SDaniel Mack 
2433f51046SDaniel Mack #define MAX9485_SCALE_256	(0 << 2)
2533f51046SDaniel Mack #define MAX9485_SCALE_384	(1 << 2)
2633f51046SDaniel Mack #define MAX9485_SCALE_768	(2 << 2)
2733f51046SDaniel Mack 
2833f51046SDaniel Mack #define MAX9485_DOUBLE		BIT(4)
2933f51046SDaniel Mack #define MAX9485_CLKOUT1_ENABLE	BIT(5)
3033f51046SDaniel Mack #define MAX9485_CLKOUT2_ENABLE	BIT(6)
3133f51046SDaniel Mack #define MAX9485_MCLK_ENABLE	BIT(7)
3233f51046SDaniel Mack #define MAX9485_FREQ_MASK	0x1f
3333f51046SDaniel Mack 
3433f51046SDaniel Mack struct max9485_rate {
3533f51046SDaniel Mack 	unsigned long out;
3633f51046SDaniel Mack 	u8 reg_value;
3733f51046SDaniel Mack };
3833f51046SDaniel Mack 
3933f51046SDaniel Mack /*
4033f51046SDaniel Mack  * Ordered by frequency. For frequency the hardware can generate with
4133f51046SDaniel Mack  * multiple settings, the one with lowest jitter is listed first.
4233f51046SDaniel Mack  */
4333f51046SDaniel Mack static const struct max9485_rate max9485_rates[] = {
4433f51046SDaniel Mack 	{  3072000, MAX9485_FS_12KHZ   | MAX9485_SCALE_256 },
4533f51046SDaniel Mack 	{  4608000, MAX9485_FS_12KHZ   | MAX9485_SCALE_384 },
4633f51046SDaniel Mack 	{  8192000, MAX9485_FS_32KHZ   | MAX9485_SCALE_256 },
4733f51046SDaniel Mack 	{  9126000, MAX9485_FS_12KHZ   | MAX9485_SCALE_768 },
4833f51046SDaniel Mack 	{ 11289600, MAX9485_FS_44_1KHZ | MAX9485_SCALE_256 },
4933f51046SDaniel Mack 	{ 12288000, MAX9485_FS_48KHZ   | MAX9485_SCALE_256 },
5033f51046SDaniel Mack 	{ 12288000, MAX9485_FS_32KHZ   | MAX9485_SCALE_384 },
5133f51046SDaniel Mack 	{ 16384000, MAX9485_FS_32KHZ   | MAX9485_SCALE_256 | MAX9485_DOUBLE },
5233f51046SDaniel Mack 	{ 16934400, MAX9485_FS_44_1KHZ | MAX9485_SCALE_384 },
5333f51046SDaniel Mack 	{ 18384000, MAX9485_FS_48KHZ   | MAX9485_SCALE_384 },
5433f51046SDaniel Mack 	{ 22579200, MAX9485_FS_44_1KHZ | MAX9485_SCALE_256 | MAX9485_DOUBLE },
5533f51046SDaniel Mack 	{ 24576000, MAX9485_FS_48KHZ   | MAX9485_SCALE_256 | MAX9485_DOUBLE },
5633f51046SDaniel Mack 	{ 24576000, MAX9485_FS_32KHZ   | MAX9485_SCALE_384 | MAX9485_DOUBLE },
5733f51046SDaniel Mack 	{ 24576000, MAX9485_FS_32KHZ   | MAX9485_SCALE_768 },
5833f51046SDaniel Mack 	{ 33868800, MAX9485_FS_44_1KHZ | MAX9485_SCALE_384 | MAX9485_DOUBLE },
5933f51046SDaniel Mack 	{ 33868800, MAX9485_FS_44_1KHZ | MAX9485_SCALE_768 },
6033f51046SDaniel Mack 	{ 36864000, MAX9485_FS_48KHZ   | MAX9485_SCALE_384 | MAX9485_DOUBLE },
6133f51046SDaniel Mack 	{ 36864000, MAX9485_FS_48KHZ   | MAX9485_SCALE_768 },
6233f51046SDaniel Mack 	{ 49152000, MAX9485_FS_32KHZ   | MAX9485_SCALE_768 | MAX9485_DOUBLE },
6333f51046SDaniel Mack 	{ 67737600, MAX9485_FS_44_1KHZ | MAX9485_SCALE_768 | MAX9485_DOUBLE },
6433f51046SDaniel Mack 	{ 73728000, MAX9485_FS_48KHZ   | MAX9485_SCALE_768 | MAX9485_DOUBLE },
6533f51046SDaniel Mack 	{ } /* sentinel */
6633f51046SDaniel Mack };
6733f51046SDaniel Mack 
6833f51046SDaniel Mack struct max9485_driver_data;
6933f51046SDaniel Mack 
7033f51046SDaniel Mack struct max9485_clk_hw {
7133f51046SDaniel Mack 	struct clk_hw hw;
7233f51046SDaniel Mack 	struct clk_init_data init;
7333f51046SDaniel Mack 	u8 enable_bit;
7433f51046SDaniel Mack 	struct max9485_driver_data *drvdata;
7533f51046SDaniel Mack };
7633f51046SDaniel Mack 
7733f51046SDaniel Mack struct max9485_driver_data {
7833f51046SDaniel Mack 	struct clk *xclk;
7933f51046SDaniel Mack 	struct i2c_client *client;
8033f51046SDaniel Mack 	u8 reg_value;
8133f51046SDaniel Mack 	struct regulator *supply;
8233f51046SDaniel Mack 	struct gpio_desc *reset_gpio;
8333f51046SDaniel Mack 	struct max9485_clk_hw hw[MAX9485_NUM_CLKS];
8433f51046SDaniel Mack };
8533f51046SDaniel Mack 
to_max9485_clk(struct clk_hw * hw)8633f51046SDaniel Mack static inline struct max9485_clk_hw *to_max9485_clk(struct clk_hw *hw)
8733f51046SDaniel Mack {
8833f51046SDaniel Mack 	return container_of(hw, struct max9485_clk_hw, hw);
8933f51046SDaniel Mack }
9033f51046SDaniel Mack 
max9485_update_bits(struct max9485_driver_data * drvdata,u8 mask,u8 value)9133f51046SDaniel Mack static int max9485_update_bits(struct max9485_driver_data *drvdata,
9233f51046SDaniel Mack 			       u8 mask, u8 value)
9333f51046SDaniel Mack {
9433f51046SDaniel Mack 	int ret;
9533f51046SDaniel Mack 
9633f51046SDaniel Mack 	drvdata->reg_value &= ~mask;
9733f51046SDaniel Mack 	drvdata->reg_value |= value;
9833f51046SDaniel Mack 
9933f51046SDaniel Mack 	dev_dbg(&drvdata->client->dev,
10033f51046SDaniel Mack 		"updating mask 0x%02x value 0x%02x -> 0x%02x\n",
10133f51046SDaniel Mack 		mask, value, drvdata->reg_value);
10233f51046SDaniel Mack 
10333f51046SDaniel Mack 	ret = i2c_master_send(drvdata->client,
10433f51046SDaniel Mack 			      &drvdata->reg_value,
10533f51046SDaniel Mack 			      sizeof(drvdata->reg_value));
10633f51046SDaniel Mack 
10733f51046SDaniel Mack 	return ret < 0 ? ret : 0;
10833f51046SDaniel Mack }
10933f51046SDaniel Mack 
max9485_clk_prepare(struct clk_hw * hw)11033f51046SDaniel Mack static int max9485_clk_prepare(struct clk_hw *hw)
11133f51046SDaniel Mack {
11233f51046SDaniel Mack 	struct max9485_clk_hw *clk_hw = to_max9485_clk(hw);
11333f51046SDaniel Mack 
11433f51046SDaniel Mack 	return max9485_update_bits(clk_hw->drvdata,
11533f51046SDaniel Mack 				   clk_hw->enable_bit,
11633f51046SDaniel Mack 				   clk_hw->enable_bit);
11733f51046SDaniel Mack }
11833f51046SDaniel Mack 
max9485_clk_unprepare(struct clk_hw * hw)11933f51046SDaniel Mack static void max9485_clk_unprepare(struct clk_hw *hw)
12033f51046SDaniel Mack {
12133f51046SDaniel Mack 	struct max9485_clk_hw *clk_hw = to_max9485_clk(hw);
12233f51046SDaniel Mack 
12333f51046SDaniel Mack 	max9485_update_bits(clk_hw->drvdata, clk_hw->enable_bit, 0);
12433f51046SDaniel Mack }
12533f51046SDaniel Mack 
12633f51046SDaniel Mack /*
12733f51046SDaniel Mack  * CLKOUT - configurable clock output
12833f51046SDaniel Mack  */
max9485_clkout_set_rate(struct clk_hw * hw,unsigned long rate,unsigned long parent_rate)12933f51046SDaniel Mack static int max9485_clkout_set_rate(struct clk_hw *hw, unsigned long rate,
13033f51046SDaniel Mack 				   unsigned long parent_rate)
13133f51046SDaniel Mack {
13233f51046SDaniel Mack 	struct max9485_clk_hw *clk_hw = to_max9485_clk(hw);
13333f51046SDaniel Mack 	const struct max9485_rate *entry;
13433f51046SDaniel Mack 
13533f51046SDaniel Mack 	for (entry = max9485_rates; entry->out != 0; entry++)
13633f51046SDaniel Mack 		if (entry->out == rate)
13733f51046SDaniel Mack 			break;
13833f51046SDaniel Mack 
13933f51046SDaniel Mack 	if (entry->out == 0)
14033f51046SDaniel Mack 		return -EINVAL;
14133f51046SDaniel Mack 
14233f51046SDaniel Mack 	return max9485_update_bits(clk_hw->drvdata,
14333f51046SDaniel Mack 				   MAX9485_FREQ_MASK,
14433f51046SDaniel Mack 				   entry->reg_value);
14533f51046SDaniel Mack }
14633f51046SDaniel Mack 
max9485_clkout_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)14733f51046SDaniel Mack static unsigned long max9485_clkout_recalc_rate(struct clk_hw *hw,
14833f51046SDaniel Mack 						unsigned long parent_rate)
14933f51046SDaniel Mack {
15033f51046SDaniel Mack 	struct max9485_clk_hw *clk_hw = to_max9485_clk(hw);
15133f51046SDaniel Mack 	struct max9485_driver_data *drvdata = clk_hw->drvdata;
15233f51046SDaniel Mack 	u8 val = drvdata->reg_value & MAX9485_FREQ_MASK;
15333f51046SDaniel Mack 	const struct max9485_rate *entry;
15433f51046SDaniel Mack 
15533f51046SDaniel Mack 	for (entry = max9485_rates; entry->out != 0; entry++)
15633f51046SDaniel Mack 		if (val == entry->reg_value)
15733f51046SDaniel Mack 			return entry->out;
15833f51046SDaniel Mack 
15933f51046SDaniel Mack 	return 0;
16033f51046SDaniel Mack }
16133f51046SDaniel Mack 
max9485_clkout_round_rate(struct clk_hw * hw,unsigned long rate,unsigned long * parent_rate)16233f51046SDaniel Mack static long max9485_clkout_round_rate(struct clk_hw *hw, unsigned long rate,
16333f51046SDaniel Mack 				      unsigned long *parent_rate)
16433f51046SDaniel Mack {
16533f51046SDaniel Mack 	const struct max9485_rate *curr, *prev = NULL;
16633f51046SDaniel Mack 
16733f51046SDaniel Mack 	for (curr = max9485_rates; curr->out != 0; curr++) {
16833f51046SDaniel Mack 		/* Exact matches */
16933f51046SDaniel Mack 		if (curr->out == rate)
17033f51046SDaniel Mack 			return rate;
17133f51046SDaniel Mack 
17233f51046SDaniel Mack 		/*
17333f51046SDaniel Mack 		 * Find the first entry that has a frequency higher than the
17433f51046SDaniel Mack 		 * requested one.
17533f51046SDaniel Mack 		 */
17633f51046SDaniel Mack 		if (curr->out > rate) {
17733f51046SDaniel Mack 			unsigned int mid;
17833f51046SDaniel Mack 
17933f51046SDaniel Mack 			/*
18033f51046SDaniel Mack 			 * If this is the first entry, clamp the value to the
18133f51046SDaniel Mack 			 * lowest possible frequency.
18233f51046SDaniel Mack 			 */
18333f51046SDaniel Mack 			if (!prev)
18433f51046SDaniel Mack 				return curr->out;
18533f51046SDaniel Mack 
18633f51046SDaniel Mack 			/*
18733f51046SDaniel Mack 			 * Otherwise, determine whether the previous entry or
18833f51046SDaniel Mack 			 * current one is closer.
18933f51046SDaniel Mack 			 */
19033f51046SDaniel Mack 			mid = prev->out + ((curr->out - prev->out) / 2);
19133f51046SDaniel Mack 
19233f51046SDaniel Mack 			return (mid > rate) ? prev->out : curr->out;
19333f51046SDaniel Mack 		}
19433f51046SDaniel Mack 
19533f51046SDaniel Mack 		prev = curr;
19633f51046SDaniel Mack 	}
19733f51046SDaniel Mack 
19833f51046SDaniel Mack 	/* If the last entry was still too high, clamp the value */
19933f51046SDaniel Mack 	return prev->out;
20033f51046SDaniel Mack }
20133f51046SDaniel Mack 
20233f51046SDaniel Mack struct max9485_clk {
20333f51046SDaniel Mack 	const char *name;
20433f51046SDaniel Mack 	int parent_index;
20533f51046SDaniel Mack 	const struct clk_ops ops;
20633f51046SDaniel Mack 	u8 enable_bit;
20733f51046SDaniel Mack };
20833f51046SDaniel Mack 
20933f51046SDaniel Mack static const struct max9485_clk max9485_clks[MAX9485_NUM_CLKS] = {
21033f51046SDaniel Mack 	[MAX9485_MCLKOUT] = {
21133f51046SDaniel Mack 		.name = "mclkout",
21233f51046SDaniel Mack 		.parent_index = -1,
21333f51046SDaniel Mack 		.enable_bit = MAX9485_MCLK_ENABLE,
21433f51046SDaniel Mack 		.ops = {
21533f51046SDaniel Mack 			.prepare	= max9485_clk_prepare,
21633f51046SDaniel Mack 			.unprepare	= max9485_clk_unprepare,
21733f51046SDaniel Mack 		},
21833f51046SDaniel Mack 	},
21933f51046SDaniel Mack 	[MAX9485_CLKOUT] = {
22033f51046SDaniel Mack 		.name = "clkout",
22133f51046SDaniel Mack 		.parent_index = -1,
22233f51046SDaniel Mack 		.ops = {
22333f51046SDaniel Mack 			.set_rate	= max9485_clkout_set_rate,
22433f51046SDaniel Mack 			.round_rate	= max9485_clkout_round_rate,
22533f51046SDaniel Mack 			.recalc_rate	= max9485_clkout_recalc_rate,
22633f51046SDaniel Mack 		},
22733f51046SDaniel Mack 	},
22833f51046SDaniel Mack 	[MAX9485_CLKOUT1] = {
22933f51046SDaniel Mack 		.name = "clkout1",
23033f51046SDaniel Mack 		.parent_index = MAX9485_CLKOUT,
23133f51046SDaniel Mack 		.enable_bit = MAX9485_CLKOUT1_ENABLE,
23233f51046SDaniel Mack 		.ops = {
23333f51046SDaniel Mack 			.prepare	= max9485_clk_prepare,
23433f51046SDaniel Mack 			.unprepare	= max9485_clk_unprepare,
23533f51046SDaniel Mack 		},
23633f51046SDaniel Mack 	},
23733f51046SDaniel Mack 	[MAX9485_CLKOUT2] = {
23833f51046SDaniel Mack 		.name = "clkout2",
23933f51046SDaniel Mack 		.parent_index = MAX9485_CLKOUT,
24033f51046SDaniel Mack 		.enable_bit = MAX9485_CLKOUT2_ENABLE,
24133f51046SDaniel Mack 		.ops = {
24233f51046SDaniel Mack 			.prepare	= max9485_clk_prepare,
24333f51046SDaniel Mack 			.unprepare	= max9485_clk_unprepare,
24433f51046SDaniel Mack 		},
24533f51046SDaniel Mack 	},
24633f51046SDaniel Mack };
24733f51046SDaniel Mack 
24833f51046SDaniel Mack static struct clk_hw *
max9485_of_clk_get(struct of_phandle_args * clkspec,void * data)24933f51046SDaniel Mack max9485_of_clk_get(struct of_phandle_args *clkspec, void *data)
25033f51046SDaniel Mack {
25133f51046SDaniel Mack 	struct max9485_driver_data *drvdata = data;
25233f51046SDaniel Mack 	unsigned int idx = clkspec->args[0];
25333f51046SDaniel Mack 
25433f51046SDaniel Mack 	return &drvdata->hw[idx].hw;
25533f51046SDaniel Mack }
25633f51046SDaniel Mack 
max9485_i2c_probe(struct i2c_client * client)25755349aedSStephen Kitt static int max9485_i2c_probe(struct i2c_client *client)
25833f51046SDaniel Mack {
25933f51046SDaniel Mack 	struct max9485_driver_data *drvdata;
26033f51046SDaniel Mack 	struct device *dev = &client->dev;
26133f51046SDaniel Mack 	const char *xclk_name;
26233f51046SDaniel Mack 	int i, ret;
26333f51046SDaniel Mack 
26433f51046SDaniel Mack 	drvdata = devm_kzalloc(dev, sizeof(*drvdata), GFP_KERNEL);
26533f51046SDaniel Mack 	if (!drvdata)
26633f51046SDaniel Mack 		return -ENOMEM;
26733f51046SDaniel Mack 
26833f51046SDaniel Mack 	drvdata->xclk = devm_clk_get(dev, "xclk");
26933f51046SDaniel Mack 	if (IS_ERR(drvdata->xclk))
27033f51046SDaniel Mack 		return PTR_ERR(drvdata->xclk);
27133f51046SDaniel Mack 
27233f51046SDaniel Mack 	xclk_name = __clk_get_name(drvdata->xclk);
27333f51046SDaniel Mack 
27433f51046SDaniel Mack 	drvdata->supply = devm_regulator_get(dev, "vdd");
27533f51046SDaniel Mack 	if (IS_ERR(drvdata->supply))
27633f51046SDaniel Mack 		return PTR_ERR(drvdata->supply);
27733f51046SDaniel Mack 
27833f51046SDaniel Mack 	ret = regulator_enable(drvdata->supply);
27933f51046SDaniel Mack 	if (ret < 0)
28033f51046SDaniel Mack 		return ret;
28133f51046SDaniel Mack 
28233f51046SDaniel Mack 	drvdata->reset_gpio =
28333f51046SDaniel Mack 		devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_HIGH);
28433f51046SDaniel Mack 	if (IS_ERR(drvdata->reset_gpio))
28533f51046SDaniel Mack 		return PTR_ERR(drvdata->reset_gpio);
28633f51046SDaniel Mack 
28733f51046SDaniel Mack 	i2c_set_clientdata(client, drvdata);
28833f51046SDaniel Mack 	drvdata->client = client;
28933f51046SDaniel Mack 
29033f51046SDaniel Mack 	ret = i2c_master_recv(drvdata->client, &drvdata->reg_value,
29133f51046SDaniel Mack 			      sizeof(drvdata->reg_value));
29233f51046SDaniel Mack 	if (ret < 0) {
29333f51046SDaniel Mack 		dev_warn(dev, "Unable to read device register: %d\n", ret);
29433f51046SDaniel Mack 		return ret;
29533f51046SDaniel Mack 	}
29633f51046SDaniel Mack 
29733f51046SDaniel Mack 	for (i = 0; i < MAX9485_NUM_CLKS; i++) {
29833f51046SDaniel Mack 		int parent_index = max9485_clks[i].parent_index;
29933f51046SDaniel Mack 		const char *name;
30033f51046SDaniel Mack 
30133f51046SDaniel Mack 		if (of_property_read_string_index(dev->of_node,
30233f51046SDaniel Mack 						  "clock-output-names",
30333f51046SDaniel Mack 						  i, &name) == 0) {
30433f51046SDaniel Mack 			drvdata->hw[i].init.name = name;
30533f51046SDaniel Mack 		} else {
30633f51046SDaniel Mack 			drvdata->hw[i].init.name = max9485_clks[i].name;
30733f51046SDaniel Mack 		}
30833f51046SDaniel Mack 
30933f51046SDaniel Mack 		drvdata->hw[i].init.ops = &max9485_clks[i].ops;
31033f51046SDaniel Mack 		drvdata->hw[i].init.num_parents = 1;
31133f51046SDaniel Mack 		drvdata->hw[i].init.flags = 0;
31233f51046SDaniel Mack 
31333f51046SDaniel Mack 		if (parent_index > 0) {
31433f51046SDaniel Mack 			drvdata->hw[i].init.parent_names =
31533f51046SDaniel Mack 				&drvdata->hw[parent_index].init.name;
31633f51046SDaniel Mack 			drvdata->hw[i].init.flags |= CLK_SET_RATE_PARENT;
31733f51046SDaniel Mack 		} else {
31833f51046SDaniel Mack 			drvdata->hw[i].init.parent_names = &xclk_name;
31933f51046SDaniel Mack 		}
32033f51046SDaniel Mack 
32133f51046SDaniel Mack 		drvdata->hw[i].enable_bit = max9485_clks[i].enable_bit;
32233f51046SDaniel Mack 		drvdata->hw[i].hw.init = &drvdata->hw[i].init;
32333f51046SDaniel Mack 		drvdata->hw[i].drvdata = drvdata;
32433f51046SDaniel Mack 
32533f51046SDaniel Mack 		ret = devm_clk_hw_register(dev, &drvdata->hw[i].hw);
32633f51046SDaniel Mack 		if (ret < 0)
32733f51046SDaniel Mack 			return ret;
32833f51046SDaniel Mack 	}
32933f51046SDaniel Mack 
33033f51046SDaniel Mack 	return devm_of_clk_add_hw_provider(dev, max9485_of_clk_get, drvdata);
33133f51046SDaniel Mack }
33233f51046SDaniel Mack 
max9485_suspend(struct device * dev)33333f51046SDaniel Mack static int __maybe_unused max9485_suspend(struct device *dev)
33433f51046SDaniel Mack {
33533f51046SDaniel Mack 	struct i2c_client *client = to_i2c_client(dev);
33633f51046SDaniel Mack 	struct max9485_driver_data *drvdata = i2c_get_clientdata(client);
33733f51046SDaniel Mack 
33833f51046SDaniel Mack 	gpiod_set_value_cansleep(drvdata->reset_gpio, 0);
33933f51046SDaniel Mack 
34033f51046SDaniel Mack 	return 0;
34133f51046SDaniel Mack }
34233f51046SDaniel Mack 
max9485_resume(struct device * dev)34333f51046SDaniel Mack static int __maybe_unused max9485_resume(struct device *dev)
34433f51046SDaniel Mack {
34533f51046SDaniel Mack 	struct i2c_client *client = to_i2c_client(dev);
34633f51046SDaniel Mack 	struct max9485_driver_data *drvdata = i2c_get_clientdata(client);
34733f51046SDaniel Mack 	int ret;
34833f51046SDaniel Mack 
34933f51046SDaniel Mack 	gpiod_set_value_cansleep(drvdata->reset_gpio, 1);
35033f51046SDaniel Mack 
35133f51046SDaniel Mack 	ret = i2c_master_send(client, &drvdata->reg_value,
35233f51046SDaniel Mack 			      sizeof(drvdata->reg_value));
35333f51046SDaniel Mack 
35433f51046SDaniel Mack 	return ret < 0 ? ret : 0;
35533f51046SDaniel Mack }
35633f51046SDaniel Mack 
35733f51046SDaniel Mack static const struct dev_pm_ops max9485_pm_ops = {
35833f51046SDaniel Mack 	SET_SYSTEM_SLEEP_PM_OPS(max9485_suspend, max9485_resume)
35933f51046SDaniel Mack };
36033f51046SDaniel Mack 
36133f51046SDaniel Mack static const struct of_device_id max9485_dt_ids[] = {
36233f51046SDaniel Mack 	{ .compatible = "maxim,max9485", },
36333f51046SDaniel Mack 	{ }
36433f51046SDaniel Mack };
36533f51046SDaniel Mack MODULE_DEVICE_TABLE(of, max9485_dt_ids);
36633f51046SDaniel Mack 
36733f51046SDaniel Mack static const struct i2c_device_id max9485_i2c_ids[] = {
36833f51046SDaniel Mack 	{ .name = "max9485", },
36933f51046SDaniel Mack 	{ }
37033f51046SDaniel Mack };
37133f51046SDaniel Mack MODULE_DEVICE_TABLE(i2c, max9485_i2c_ids);
37233f51046SDaniel Mack 
37333f51046SDaniel Mack static struct i2c_driver max9485_driver = {
37433f51046SDaniel Mack 	.driver = {
37533f51046SDaniel Mack 		.name		= "max9485",
37633f51046SDaniel Mack 		.pm		= &max9485_pm_ops,
37733f51046SDaniel Mack 		.of_match_table	= max9485_dt_ids,
37833f51046SDaniel Mack 	},
379*62279db5SUwe Kleine-König 	.probe = max9485_i2c_probe,
38033f51046SDaniel Mack 	.id_table = max9485_i2c_ids,
38133f51046SDaniel Mack };
38233f51046SDaniel Mack module_i2c_driver(max9485_driver);
38333f51046SDaniel Mack 
38433f51046SDaniel Mack MODULE_AUTHOR("Daniel Mack <daniel@zonque.org>");
38533f51046SDaniel Mack MODULE_DESCRIPTION("MAX9485 Programmable Audio Clock Generator");
38633f51046SDaniel Mack MODULE_LICENSE("GPL v2");
387