cregit-Linux how code gets into the kernel

Release 4.11 drivers/staging/iio/impedance-analyzer/ad5933.c

/*
 * AD5933 AD5934 Impedance Converter, Network Analyzer
 *
 * Copyright 2011 Analog Devices Inc.
 *
 * Licensed under the GPL-2.
 */

#include <linux/interrupt.h>
#include <linux/device.h>
#include <linux/kernel.h>
#include <linux/sysfs.h>
#include <linux/i2c.h>
#include <linux/regulator/consumer.h>
#include <linux/types.h>
#include <linux/err.h>
#include <linux/delay.h>
#include <linux/module.h>

#include <linux/iio/iio.h>
#include <linux/iio/sysfs.h>
#include <linux/iio/buffer.h>
#include <linux/iio/kfifo_buf.h>

/* AD5933/AD5934 Registers */

#define AD5933_REG_CONTROL_HB		0x80	
/* R/W, 1 byte */

#define AD5933_REG_CONTROL_LB		0x81	
/* R/W, 1 byte */

#define AD5933_REG_FREQ_START		0x82	
/* R/W, 3 bytes */

#define AD5933_REG_FREQ_INC		0x85	
/* R/W, 3 bytes */

#define AD5933_REG_INC_NUM		0x88	
/* R/W, 2 bytes, 9 bit */

#define AD5933_REG_SETTLING_CYCLES	0x8A	
/* R/W, 2 bytes */

#define AD5933_REG_STATUS		0x8F	
/* R, 1 byte */

#define AD5933_REG_TEMP_DATA		0x92	
/* R, 2 bytes*/

#define AD5933_REG_REAL_DATA		0x94	
/* R, 2 bytes*/

#define AD5933_REG_IMAG_DATA		0x96	
/* R, 2 bytes*/

/* AD5933_REG_CONTROL_HB Bits */

#define AD5933_CTRL_INIT_START_FREQ	(0x1 << 4)

#define AD5933_CTRL_START_SWEEP		(0x2 << 4)

#define AD5933_CTRL_INC_FREQ		(0x3 << 4)

#define AD5933_CTRL_REPEAT_FREQ		(0x4 << 4)

#define AD5933_CTRL_MEASURE_TEMP	(0x9 << 4)

#define AD5933_CTRL_POWER_DOWN		(0xA << 4)

#define AD5933_CTRL_STANDBY		(0xB << 4)


#define AD5933_CTRL_RANGE_2000mVpp	(0x0 << 1)

#define AD5933_CTRL_RANGE_200mVpp	(0x1 << 1)

#define AD5933_CTRL_RANGE_400mVpp	(0x2 << 1)

#define AD5933_CTRL_RANGE_1000mVpp	(0x3 << 1)

#define AD5933_CTRL_RANGE(x)		((x) << 1)


#define AD5933_CTRL_PGA_GAIN_1		(0x1 << 0)

#define AD5933_CTRL_PGA_GAIN_5		(0x0 << 0)

/* AD5933_REG_CONTROL_LB Bits */

#define AD5933_CTRL_RESET		(0x1 << 4)

#define AD5933_CTRL_INT_SYSCLK		(0x0 << 3)

#define AD5933_CTRL_EXT_SYSCLK		(0x1 << 3)

/* AD5933_REG_STATUS Bits */

#define AD5933_STAT_TEMP_VALID		(0x1 << 0)

#define AD5933_STAT_DATA_VALID		(0x1 << 1)

#define AD5933_STAT_SWEEP_DONE		(0x1 << 2)

/* I2C Block Commands */

#define AD5933_I2C_BLOCK_WRITE		0xA0

#define AD5933_I2C_BLOCK_READ		0xA1

#define AD5933_I2C_ADDR_POINTER		0xB0

/* Device Specs */

#define AD5933_INT_OSC_FREQ_Hz		16776000

#define AD5933_MAX_OUTPUT_FREQ_Hz	100000

#define AD5933_MAX_RETRIES		100


#define AD5933_OUT_RANGE		1

#define AD5933_OUT_RANGE_AVAIL		2

#define AD5933_OUT_SETTLING_CYCLES	3

#define AD5933_IN_PGA_GAIN		4

#define AD5933_IN_PGA_GAIN_AVAIL	5

#define AD5933_FREQ_POINTS		6


#define AD5933_POLL_TIME_ms		10

#define AD5933_INIT_EXCITATION_TIME_ms	100

/**
 * struct ad5933_platform_data - platform specific data
 * @ext_clk_Hz:         the external clock frequency in Hz, if not set
 *                      the driver uses the internal clock (16.776 MHz)
 * @vref_mv:            the external reference voltage in millivolt
 */


struct ad5933_platform_data {
	
unsigned long			ext_clk_Hz;
	
unsigned short			vref_mv;
};


struct ad5933_state {
	
struct i2c_client		*client;
	
struct regulator		*reg;
	
struct delayed_work		work;
	
unsigned long			mclk_hz;
	
unsigned char			ctrl_hb;
	
unsigned char			ctrl_lb;
	
unsigned int			range_avail[4];
	
unsigned short			vref_mv;
	
unsigned short			settling_cycles;
	
unsigned short			freq_points;
	
unsigned int			freq_start;
	
unsigned int			freq_inc;
	
unsigned int			state;
	
unsigned int			poll_time_jiffies;
};


static struct ad5933_platform_data ad5933_default_pdata  = {
	.vref_mv = 3300,
};


static const struct iio_chan_spec ad5933_channels[] = {
	{
		.type = IIO_TEMP,
		.indexed = 1,
		.channel = 0,
		.info_mask_separate = BIT(IIO_CHAN_INFO_RAW) |
			BIT(IIO_CHAN_INFO_SCALE),
		.address = AD5933_REG_TEMP_DATA,
		.scan_index = -1,
		.scan_type = {
			.sign = 's',
			.realbits = 14,
			.storagebits = 16,
                },
        }, { /* Ring Channels */
		.type = IIO_VOLTAGE,
		.indexed = 1,
		.channel = 0,
		.extend_name = "real",
		.address = AD5933_REG_REAL_DATA,
		.scan_index = 0,
		.scan_type = {
			.sign = 's',
			.realbits = 16,
			.storagebits = 16,
                },
        }, {
		.type = IIO_VOLTAGE,
		.indexed = 1,
		.channel = 0,
		.extend_name = "imag",
		.address = AD5933_REG_IMAG_DATA,
		.scan_index = 1,
		.scan_type = {
			.sign = 's',
			.realbits = 16,
			.storagebits = 16,
                },
        },
};


static int ad5933_i2c_write(struct i2c_client *client, u8 reg, u8 len, u8 *data) { int ret; while (len--) { ret = i2c_smbus_write_byte_data(client, reg++, *data++); if (ret < 0) { dev_err(&client->dev, "I2C write error\n"); return ret; } } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich69100.00%1100.00%
Total69100.00%1100.00%


static int ad5933_i2c_read(struct i2c_client *client, u8 reg, u8 len, u8 *data) { int ret; while (len--) { ret = i2c_smbus_read_byte_data(client, reg++); if (ret < 0) { dev_err(&client->dev, "I2C read error\n"); return ret; } *data++ = ret; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich71100.00%1100.00%
Total71100.00%1100.00%


static int ad5933_cmd(struct ad5933_state *st, unsigned char cmd) { unsigned char dat = st->ctrl_hb | cmd; return ad5933_i2c_write(st->client, AD5933_REG_CONTROL_HB, 1, &dat); }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich40100.00%1100.00%
Total40100.00%1100.00%


static int ad5933_reset(struct ad5933_state *st) { unsigned char dat = st->ctrl_lb | AD5933_CTRL_RESET; return ad5933_i2c_write(st->client, AD5933_REG_CONTROL_LB, 1, &dat); }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich36100.00%1100.00%
Total36100.00%1100.00%


static int ad5933_wait_busy(struct ad5933_state *st, unsigned char event) { unsigned char val, timeout = AD5933_MAX_RETRIES; int ret; while (timeout--) { ret = ad5933_i2c_read(st->client, AD5933_REG_STATUS, 1, &val); if (ret < 0) return ret; if (val & event) return val; cpu_relax(); mdelay(1); } return -EAGAIN; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich79100.00%1100.00%
Total79100.00%1100.00%


static int ad5933_set_freq(struct ad5933_state *st, unsigned int reg, unsigned long freq) { unsigned long long freqreg; union { __be32 d32; u8 d8[4]; } dat; freqreg = (u64) freq * (u64) (1 << 27); do_div(freqreg, st->mclk_hz / 4); switch (reg) { case AD5933_REG_FREQ_START: st->freq_start = freq; break; case AD5933_REG_FREQ_INC: st->freq_inc = freq; break; default: return -EINVAL; } dat.d32 = cpu_to_be32(freqreg); return ad5933_i2c_write(st->client, reg, 3, &dat.d8[1]); }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich12398.40%133.33%
Teodora Baluta10.80%133.33%
Alison Schofield10.80%133.33%
Total125100.00%3100.00%


static int ad5933_setup(struct ad5933_state *st) { __be16 dat; int ret; ret = ad5933_reset(st); if (ret < 0) return ret; ret = ad5933_set_freq(st, AD5933_REG_FREQ_START, 10000); if (ret < 0) return ret; ret = ad5933_set_freq(st, AD5933_REG_FREQ_INC, 200); if (ret < 0) return ret; st->settling_cycles = 10; dat = cpu_to_be16(st->settling_cycles); ret = ad5933_i2c_write(st->client, AD5933_REG_SETTLING_CYCLES, 2, (u8 *)&dat); if (ret < 0) return ret; st->freq_points = 100; dat = cpu_to_be16(st->freq_points); return ad5933_i2c_write(st->client, AD5933_REG_INC_NUM, 2, (u8 *)&dat); }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich15099.34%150.00%
Teodora Baluta10.66%150.00%
Total151100.00%2100.00%


static void ad5933_calc_out_ranges(struct ad5933_state *st) { int i; unsigned int normalized_3v3[4] = {1980, 198, 383, 970}; for (i = 0; i < 4; i++) st->range_avail[i] = normalized_3v3[i] * st->vref_mv / 3300; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich6198.39%150.00%
Alison Schofield11.61%150.00%
Total62100.00%2100.00%

/* * handles: AD5933_REG_FREQ_START and AD5933_REG_FREQ_INC */
static ssize_t ad5933_show_frequency(struct device *dev, struct device_attribute *attr, char *buf) { struct iio_dev *indio_dev = dev_to_iio_dev(dev); struct ad5933_state *st = iio_priv(indio_dev); struct iio_dev_attr *this_attr = to_iio_dev_attr(attr); int ret; unsigned long long freqreg; union { __be32 d32; u8 d8[4]; } dat; mutex_lock(&indio_dev->mlock); ret = ad5933_i2c_read(st->client, this_attr->address, 3, &dat.d8[1]); mutex_unlock(&indio_dev->mlock); if (ret < 0) return ret; freqreg = be32_to_cpu(dat.d32) & 0xFFFFFF; freqreg = (u64)freqreg * (u64)(st->mclk_hz / 4); do_div(freqreg, 1 << 27); return sprintf(buf, "%d\n", (int)freqreg); }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich16596.49%125.00%
Jonathan Cameron42.34%125.00%
Lars-Peter Clausen10.58%125.00%
Teodora Baluta10.58%125.00%
Total171100.00%4100.00%


static ssize_t ad5933_store_frequency(struct device *dev, struct device_attribute *attr, const char *buf, size_t len) { struct iio_dev *indio_dev = dev_to_iio_dev(dev); struct ad5933_state *st = iio_priv(indio_dev); struct iio_dev_attr *this_attr = to_iio_dev_attr(attr); unsigned long val; int ret; ret = kstrtoul(buf, 10, &val); if (ret) return ret; if (val > AD5933_MAX_OUTPUT_FREQ_Hz) return -EINVAL; mutex_lock(&indio_dev->mlock); ret = ad5933_set_freq(st, this_attr->address, val); mutex_unlock(&indio_dev->mlock); return ret ? ret : len; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich11994.44%125.00%
Jonathan Cameron43.17%125.00%
Jingoo Han21.59%125.00%
Lars-Peter Clausen10.79%125.00%
Total126100.00%4100.00%

static IIO_DEVICE_ATTR(out_voltage0_freq_start, S_IRUGO | S_IWUSR, ad5933_show_frequency, ad5933_store_frequency, AD5933_REG_FREQ_START); static IIO_DEVICE_ATTR(out_voltage0_freq_increment, S_IRUGO | S_IWUSR, ad5933_show_frequency, ad5933_store_frequency, AD5933_REG_FREQ_INC);
static ssize_t ad5933_show(struct device *dev, struct device_attribute *attr, char *buf) { struct iio_dev *indio_dev = dev_to_iio_dev(dev); struct ad5933_state *st = iio_priv(indio_dev); struct iio_dev_attr *this_attr = to_iio_dev_attr(attr); int ret = 0, len = 0; mutex_lock(&indio_dev->mlock); switch ((u32)this_attr->address) { case AD5933_OUT_RANGE: len = sprintf(buf, "%u\n", st->range_avail[(st->ctrl_hb >> 1) & 0x3]); break; case AD5933_OUT_RANGE_AVAIL: len = sprintf(buf, "%u %u %u %u\n", st->range_avail[0], st->range_avail[3], st->range_avail[2], st->range_avail[1]); break; case AD5933_OUT_SETTLING_CYCLES: len = sprintf(buf, "%d\n", st->settling_cycles); break; case AD5933_IN_PGA_GAIN: len = sprintf(buf, "%s\n", (st->ctrl_hb & AD5933_CTRL_PGA_GAIN_1) ? "1" : "0.2"); break; case AD5933_IN_PGA_GAIN_AVAIL: len = sprintf(buf, "1 0.2\n"); break; case AD5933_FREQ_POINTS: len = sprintf(buf, "%d\n", st->freq_points); break; default: ret = -EINVAL; } mutex_unlock(&indio_dev->mlock); return ret ? ret : len; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich23397.08%240.00%
Jonathan Cameron41.67%120.00%
Asaf Vertz20.83%120.00%
Lars-Peter Clausen10.42%120.00%
Total240100.00%5100.00%


static ssize_t ad5933_store(struct device *dev, struct device_attribute *attr, const char *buf, size_t len) { struct iio_dev *indio_dev = dev_to_iio_dev(dev); struct ad5933_state *st = iio_priv(indio_dev); struct iio_dev_attr *this_attr = to_iio_dev_attr(attr); u16 val; int i, ret = 0; __be16 dat; if (this_attr->address != AD5933_IN_PGA_GAIN) { ret = kstrtou16(buf, 10, &val); if (ret) return ret; } mutex_lock(&indio_dev->mlock); switch ((u32)this_attr->address) { case AD5933_OUT_RANGE: ret = -EINVAL; for (i = 0; i < 4; i++) if (val == st->range_avail[i]) { st->ctrl_hb &= ~AD5933_CTRL_RANGE(0x3); st->ctrl_hb |= AD5933_CTRL_RANGE(i); ret = ad5933_cmd(st, 0); break; } break; case AD5933_IN_PGA_GAIN: if (sysfs_streq(buf, "1")) { st->ctrl_hb |= AD5933_CTRL_PGA_GAIN_1; } else if (sysfs_streq(buf, "0.2")) { st->ctrl_hb &= ~AD5933_CTRL_PGA_GAIN_1; } else { ret = -EINVAL; break; } ret = ad5933_cmd(st, 0); break; case AD5933_OUT_SETTLING_CYCLES: val = clamp(val, (u16)0, (u16)0x7FF); st->settling_cycles = val; /* 2x, 4x handling, see datasheet */ if (val > 1022) val = (val >> 2) | (3 << 9); else if (val > 511) val = (val >> 1) | (1 << 9); dat = cpu_to_be16(val); ret = ad5933_i2c_write(st->client, AD5933_REG_SETTLING_CYCLES, 2, (u8 *)&dat); break; case AD5933_FREQ_POINTS: val = clamp(val, (u16)0, (u16)511); st->freq_points = val; dat = cpu_to_be16(val); ret = ad5933_i2c_write(st->client, AD5933_REG_INC_NUM, 2, (u8 *)&dat); break; default: ret = -EINVAL; } mutex_unlock(&indio_dev->mlock); return ret ? ret : len; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich37391.42%225.00%
Jingoo Han184.41%112.50%
Luis de Bethencourt61.47%112.50%
Phil Turnbull51.23%112.50%
Jonathan Cameron40.98%112.50%
Lars-Peter Clausen10.25%112.50%
Teodora Baluta10.25%112.50%
Total408100.00%8100.00%

static IIO_DEVICE_ATTR(out_voltage0_scale, S_IRUGO | S_IWUSR, ad5933_show, ad5933_store, AD5933_OUT_RANGE); static IIO_DEVICE_ATTR(out_voltage0_scale_available, S_IRUGO, ad5933_show, NULL, AD5933_OUT_RANGE_AVAIL); static IIO_DEVICE_ATTR(in_voltage0_scale, S_IRUGO | S_IWUSR, ad5933_show, ad5933_store, AD5933_IN_PGA_GAIN); static IIO_DEVICE_ATTR(in_voltage0_scale_available, S_IRUGO, ad5933_show, NULL, AD5933_IN_PGA_GAIN_AVAIL); static IIO_DEVICE_ATTR(out_voltage0_freq_points, S_IRUGO | S_IWUSR, ad5933_show, ad5933_store, AD5933_FREQ_POINTS); static IIO_DEVICE_ATTR(out_voltage0_settling_cycles, S_IRUGO | S_IWUSR, ad5933_show, ad5933_store, AD5933_OUT_SETTLING_CYCLES); /* note: * ideally we would handle the scale attributes via the iio_info * (read|write)_raw methods, however this part is a untypical since we * don't create dedicated sysfs channel attributes for out0 and in0. */ static struct attribute *ad5933_attributes[] = { &iio_dev_attr_out_voltage0_scale.dev_attr.attr, &iio_dev_attr_out_voltage0_scale_available.dev_attr.attr, &iio_dev_attr_out_voltage0_freq_start.dev_attr.attr, &iio_dev_attr_out_voltage0_freq_increment.dev_attr.attr, &iio_dev_attr_out_voltage0_freq_points.dev_attr.attr, &iio_dev_attr_out_voltage0_settling_cycles.dev_attr.attr, &iio_dev_attr_in_voltage0_scale.dev_attr.attr, &iio_dev_attr_in_voltage0_scale_available.dev_attr.attr, NULL }; static const struct attribute_group ad5933_attribute_group = { .attrs = ad5933_attributes, };
static int ad5933_read_raw(struct iio_dev *indio_dev, struct iio_chan_spec const *chan, int *val, int *val2, long m) { struct ad5933_state *st = iio_priv(indio_dev); __be16 dat; int ret; switch (m) { case IIO_CHAN_INFO_RAW: mutex_lock(&indio_dev->mlock); if (iio_buffer_enabled(indio_dev)) { ret = -EBUSY; goto out; } ret = ad5933_cmd(st, AD5933_CTRL_MEASURE_TEMP); if (ret < 0) goto out; ret = ad5933_wait_busy(st, AD5933_STAT_TEMP_VALID); if (ret < 0) goto out; ret = ad5933_i2c_read(st->client, AD5933_REG_TEMP_DATA, 2, (u8 *)&dat); if (ret < 0) goto out; mutex_unlock(&indio_dev->mlock); *val = sign_extend32(be16_to_cpu(dat), 13); return IIO_VAL_INT; case IIO_CHAN_INFO_SCALE: *val = 1000; *val2 = 5; return IIO_VAL_FRACTIONAL_LOG2; } return -EINVAL; out: mutex_unlock(&indio_dev->mlock); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich16381.50%116.67%
Lars-Peter Clausen2814.00%116.67%
Jonathan Cameron84.00%350.00%
Teodora Baluta10.50%116.67%
Total200100.00%6100.00%

static const struct iio_info ad5933_info = { .read_raw = ad5933_read_raw, .attrs = &ad5933_attribute_group, .driver_module = THIS_MODULE, };
static int ad5933_ring_preenable(struct iio_dev *indio_dev) { struct ad5933_state *st = iio_priv(indio_dev); int ret; if (bitmap_empty(indio_dev->active_scan_mask, indio_dev->masklength)) return -EINVAL; ret = ad5933_reset(st); if (ret < 0) return ret; ret = ad5933_cmd(st, AD5933_CTRL_STANDBY); if (ret < 0) return ret; ret = ad5933_cmd(st, AD5933_CTRL_INIT_START_FREQ); if (ret < 0) return ret; st->state = AD5933_CTRL_INIT_START_FREQ; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich9290.20%133.33%
Jonathan Cameron109.80%266.67%
Total102100.00%3100.00%


static int ad5933_ring_postenable(struct iio_dev *indio_dev) { struct ad5933_state *st = iio_priv(indio_dev); /* AD5933_CTRL_INIT_START_FREQ: * High Q complex circuits require a long time to reach steady state. * To facilitate the measurement of such impedances, this mode allows * the user full control of the settling time requirement before * entering start frequency sweep mode where the impedance measurement * takes place. In this mode the impedance is excited with the * programmed start frequency (ad5933_ring_preenable), * but no measurement takes place. */ schedule_delayed_work(&st->work, msecs_to_jiffies(AD5933_INIT_EXCITATION_TIME_ms)); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich38100.00%1100.00%
Total38100.00%1100.00%


static int ad5933_ring_postdisable(struct iio_dev *indio_dev) { struct ad5933_state *st = iio_priv(indio_dev); cancel_delayed_work_sync(&st->work); return ad5933_cmd(st, AD5933_CTRL_POWER_DOWN); }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich37100.00%1100.00%
Total37100.00%1100.00%

static const struct iio_buffer_setup_ops ad5933_ring_setup_ops = { .preenable = ad5933_ring_preenable, .postenable = ad5933_ring_postenable, .postdisable = ad5933_ring_postdisable, };
static int ad5933_register_ring_funcs_and_init(struct iio_dev *indio_dev) { struct iio_buffer *buffer; buffer = iio_kfifo_allocate(); if (!buffer) return -ENOMEM; iio_device_attach_buffer(indio_dev, buffer); /* Ring buffer functions - here trigger setup related */ indio_dev->setup_ops = &ad5933_ring_setup_ops; indio_dev->modes |= INDIO_BUFFER_HARDWARE; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich3666.67%114.29%
Lars-Peter Clausen1222.22%114.29%
Jonathan Cameron47.41%342.86%
Julia Lawall11.85%114.29%
Karol Wrona11.85%114.29%
Total54100.00%7100.00%


static void ad5933_work(struct work_struct *work) { struct ad5933_state *st = container_of(work, struct ad5933_state, work.work); struct iio_dev *indio_dev = i2c_get_clientdata(st->client); __be16 buf[2]; int val[2]; unsigned char status; int ret; mutex_lock(&indio_dev->mlock); if (st->state == AD5933_CTRL_INIT_START_FREQ) { /* start sweep */ ad5933_cmd(st, AD5933_CTRL_START_SWEEP); st->state = AD5933_CTRL_START_SWEEP; schedule_delayed_work(&st->work, st->poll_time_jiffies); goto out; } ret = ad5933_i2c_read(st->client, AD5933_REG_STATUS, 1, &status); if (ret) goto out; if (status & AD5933_STAT_DATA_VALID) { int scan_count = bitmap_weight(indio_dev->active_scan_mask, indio_dev->masklength); ret = ad5933_i2c_read(st->client, test_bit(1, indio_dev->active_scan_mask) ? AD5933_REG_REAL_DATA : AD5933_REG_IMAG_DATA, scan_count * 2, (u8 *)buf); if (ret) goto out; if (scan_count == 2) { val[0] = be16_to_cpu(buf[0]); val[1] = be16_to_cpu(buf[1]); } else { val[0] = be16_to_cpu(buf[0]); } iio_push_to_buffers(indio_dev, val); } else { /* no data available - try again later */ schedule_delayed_work(&st->work, st->poll_time_jiffies); goto out; } if (status & AD5933_STAT_SWEEP_DONE) { /* last sample received - power down do * nothing until the ring enable is toggled */ ad5933_cmd(st, AD5933_CTRL_POWER_DOWN); } else { /* we just received a valid datum, move on to the next */ ad5933_cmd(st, AD5933_CTRL_INC_FREQ); schedule_delayed_work(&st->work, st->poll_time_jiffies); } out: mutex_unlock(&indio_dev->mlock); }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich25179.68%114.29%
Arnd Bergmann299.21%114.29%
Jonathan Cameron237.30%342.86%
Ksenija Stanojevic113.49%114.29%
Alison Schofield10.32%114.29%
Total315100.00%7100.00%


static int ad5933_probe(struct i2c_client *client, const struct i2c_device_id *id) { int ret, voltage_uv = 0; struct ad5933_platform_data *pdata = dev_get_platdata(&client->dev); struct ad5933_state *st; struct iio_dev *indio_dev; indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*st)); if (!indio_dev) return -ENOMEM; st = iio_priv(indio_dev); i2c_set_clientdata(client, indio_dev); st->client = client; if (!pdata) pdata = &ad5933_default_pdata; st->reg = devm_regulator_get(&client->dev, "vdd"); if (IS_ERR(st->reg)) return PTR_ERR(st->reg); ret = regulator_enable(st->reg); if (ret) { dev_err(&client->dev, "Failed to enable specified VDD supply\n"); return ret; } voltage_uv = regulator_get_voltage(st->reg); if (voltage_uv) st->vref_mv = voltage_uv / 1000; else st->vref_mv = pdata->vref_mv; if (pdata->ext_clk_Hz) { st->mclk_hz = pdata->ext_clk_Hz; st->ctrl_lb = AD5933_CTRL_EXT_SYSCLK; } else { st->mclk_hz = AD5933_INT_OSC_FREQ_Hz; st->ctrl_lb = AD5933_CTRL_INT_SYSCLK; } ad5933_calc_out_ranges(st); INIT_DELAYED_WORK(&st->work, ad5933_work); st->poll_time_jiffies = msecs_to_jiffies(AD5933_POLL_TIME_ms); indio_dev->dev.parent = &client->dev; indio_dev->info = &ad5933_info; indio_dev->name = id->name; indio_dev->modes = INDIO_DIRECT_MODE; indio_dev->channels = ad5933_channels; indio_dev->num_channels = ARRAY_SIZE(ad5933_channels); ret = ad5933_register_ring_funcs_and_init(indio_dev); if (ret) goto error_disable_reg; ret = ad5933_setup(st); if (ret) goto error_unreg_ring; ret = iio_device_register(indio_dev); if (ret) goto error_unreg_ring; return 0; error_unreg_ring: iio_kfifo_free(indio_dev->buffer); error_disable_reg: regulator_disable(st->reg); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich30483.29%19.09%
Eva Rachel Retuya215.75%218.18%
Jonathan Cameron154.11%327.27%
Sachin Kamat143.84%19.09%
Lars-Peter Clausen61.64%218.18%
Alison Schofield41.10%19.09%
Cristina Opriceana10.27%19.09%
Total365100.00%11100.00%


static int ad5933_remove(struct i2c_client *client) { struct iio_dev *indio_dev = i2c_get_clientdata(client); struct ad5933_state *st = iio_priv(indio_dev); iio_device_unregister(indio_dev); iio_kfifo_free(indio_dev->buffer); regulator_disable(st->reg); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich4686.79%125.00%
Jonathan Cameron713.21%375.00%
Total53100.00%4100.00%

static const struct i2c_device_id ad5933_id[] = { { "ad5933", 0 }, { "ad5934", 0 }, {} }; MODULE_DEVICE_TABLE(i2c, ad5933_id); static struct i2c_driver ad5933_driver = { .driver = { .name = "ad5933", }, .probe = ad5933_probe, .remove = ad5933_remove, .id_table = ad5933_id, }; module_i2c_driver(ad5933_driver); MODULE_AUTHOR("Michael Hennerich <hennerich@blackfin.uclinux.org>"); MODULE_DESCRIPTION("Analog Devices AD5933 Impedance Conv. Network Analyzer"); MODULE_LICENSE("GPL v2");

Overall Contributors

PersonTokensPropCommitsCommitProp
Michael Hennerich313887.60%24.44%
Jonathan Cameron2316.45%1635.56%
Lars-Peter Clausen661.84%817.78%
Arnd Bergmann290.81%12.22%
Alison Schofield260.73%48.89%
Eva Rachel Retuya210.59%24.44%
Jingoo Han200.56%12.22%
Sachin Kamat140.39%12.22%
Ksenija Stanojevic110.31%12.22%
Luis de Bethencourt60.17%12.22%
Phil Turnbull50.14%12.22%
Teodora Baluta50.14%12.22%
Paul Gortmaker30.08%12.22%
Nizam Haider20.06%12.22%
Asaf Vertz20.06%12.22%
Julia Lawall10.03%12.22%
Karol Wrona10.03%12.22%
Cristina Opriceana10.03%12.22%
Total3582100.00%45100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.