Initial Contribution

msm-2.6.38: tag AU_LINUX_ANDROID_GINGERBREAD.02.03.04.00.142

Signed-off-by: Bryan Huntsman <bryanh@codeaurora.org>
diff --git a/include/linux/msm_adc.h b/include/linux/msm_adc.h
new file mode 100644
index 0000000..b8ab599
--- /dev/null
+++ b/include/linux/msm_adc.h
@@ -0,0 +1,347 @@
+#ifndef __MSM_ADC_H
+#define __MSM_ADC_H
+
+#include <linux/sched.h>
+
+#define MSM_ADC_MAX_CHAN_STR 64
+
+/* must be <= to the max buffer size in the modem implementation */
+#define MSM_ADC_DEV_MAX_INFLIGHT 9
+
+#define MSM_ADC_IOCTL_CODE		0x90
+
+struct msm_adc_conversion {
+	/* hwmon channel number - this is not equivalent to the DAL chan */
+	uint32_t chan;
+	/* returned result in ms */
+	int result;
+};
+
+struct adc_chan_result {
+	/* The channel number of the requesting/requested conversion */
+	uint32_t chan;
+	/* The pre-calibrated digital output of a given ADC relative to the
+	ADC reference */
+	int32_t adc_code;
+	/* in units specific for a given ADC; most ADC uses reference voltage
+	 *  but some ADC uses reference current.  This measurement here is
+	 *  a number relative to a reference of a given ADC */
+	int64_t measurement;
+	/* The data meaningful for each individual channel whether it is
+	 * voltage, current, temperature, etc. */
+	int64_t physical;
+};
+
+/*
+ * Issue a blocking adc conversion request. Once the call returns, the data
+ * can be found in the 'physical' field of adc_chan_result. This call will
+ * return ENODATA if there is an invalid result returned by the modem driver.
+ */
+#define MSM_ADC_REQUEST			_IOWR(MSM_ADC_IOCTL_CODE, 1,	\
+					     struct adc_chan_result)
+
+/*
+ * Issue a non-blocking adc conversion request. The results from this
+ * request can be obtained by calling AIO_READ once the transfer is
+ * completed. To verify completion, the blocking call AIO_POLL can be used.
+ * If there are no slot resources, this call will return an error with errno
+ * set to EWOULDBLOCK.
+ */
+#define MSM_ADC_AIO_REQUEST		_IOWR(MSM_ADC_IOCTL_CODE, 2,	\
+					     struct adc_chan_result)
+
+/*
+ * Same non-blocking semantics as AIO_REQUEST, except this call will block
+ * if there are no available slot resources. This call can fail with errno
+ * set to EDEADLK if there are no resources and the file descriptor in question
+ * has outstanding conversion requests already. This is done so the client
+ * does not block on resources that can only be freed by reading the results --
+ * effectively deadlocking the system. In this case, the client must read
+ * pending results before proceeding to free up resources.
+ */
+#define MSM_ADC_AIO_REQUEST_BLOCK_RES	_IOWR(MSM_ADC_IOCTL_CODE, 3,	\
+					     struct adc_chan_result)
+
+/*
+ * Returns the number of pending results that are associated with a particular
+ * file descriptor. If there are no pending results, this call will block until
+ * there is at least one. If there are no requests queued at all on this file
+ * descriptor, this call will fail with EDEADLK. This is to prevent deadlock in
+ * a single-threaded scenario where POLL would never return.
+ */
+#define MSM_ADC_AIO_POLL		_IOR(MSM_ADC_IOCTL_CODE, 4,	\
+					     uint32_t)
+
+#define MSM_ADC_FLUID_INIT	_IOR(MSM_ADC_IOCTL_CODE, 5,	\
+					     uint32_t)
+
+#define MSM_ADC_FLUID_DEINIT	_IOR(MSM_ADC_IOCTL_CODE, 6,	\
+					     uint32_t)
+
+struct msm_adc_aio_result {
+	uint32_t chan;
+	int result;
+};
+
+/*
+ * Read the results from an AIO / non-blocking conversion request. AIO_POLL
+ * should be used before using this command to verify how many pending requests
+ * are available for the file descriptor. This call will fail with errno set to
+ * ENOMSG if there are no pending messages to be read at the time of the call.
+ * The call will return ENODATA if there is an invalid result returned by the
+ * modem driver.
+ */
+#define MSM_ADC_AIO_READ		_IOR(MSM_ADC_IOCTL_CODE, 5,	\
+					     struct adc_chan_result)
+
+struct msm_adc_lookup {
+	/* channel name (input) */
+	char name[MSM_ADC_MAX_CHAN_STR];
+	/* local channel index (output) */
+	uint32_t chan_idx;
+};
+
+/*
+ * Look up a channel name and get back an index that can be used
+ * as a parameter to the conversion request commands.
+ */
+#define MSM_ADC_LOOKUP			_IOWR(MSM_ADC_IOCTL_CODE, 6,	\
+					     struct msm_adc_lookup)
+
+
+#ifdef __KERNEL__
+#define MSM_ADC_MAX_NUM_DEVS		3
+
+enum {
+	ADC_CONFIG_TYPE1,
+	ADC_CONFIG_TYPE2,
+	ADC_CONFIG_NONE = 0xffffffff
+};
+
+enum {
+	ADC_CALIB_CONFIG_TYPE1,
+	ADC_CALIB_CONFIG_TYPE2,
+	ADC_CALIB_CONFIG_TYPE3,
+	ADC_CALIB_CONFIG_TYPE4,
+	ADC_CALIB_CONFIG_TYPE5,
+	ADC_CALIB_CONFIG_TYPE6,
+	ADC_CALIB_CONFIG_TYPE7,
+	ADC_CALIB_CONFIG_NONE = 0xffffffff
+};
+
+enum {
+	/* CHAN_PATH_TYPEn is specific for each ADC driver
+	and can be used however way it wants*/
+	CHAN_PATH_TYPE1,
+	CHAN_PATH_TYPE2,
+	CHAN_PATH_TYPE3,
+	CHAN_PATH_TYPE4,
+	CHAN_PATH_TYPE5,
+	CHAN_PATH_TYPE6,
+	CHAN_PATH_TYPE7,
+	CHAN_PATH_TYPE8,
+	CHAN_PATH_TYPE9,
+	CHAN_PATH_TYPE10,
+	CHAN_PATH_TYPE11,
+	CHAN_PATH_TYPE12,
+	CHAN_PATH_TYPE13,
+	CHAN_PATH_TYPE14,
+	CHAN_PATH_TYPE15,
+	CHAN_PATH_TYPE16,
+	/* A given channel connects directly to the ADC */
+	CHAN_PATH_TYPE_NONE = 0xffffffff
+};
+
+#define CHANNEL_ADC_BATT_ID     0
+#define CHANNEL_ADC_BATT_THERM  1
+#define CHANNEL_ADC_BATT_AMON   2
+#define CHANNEL_ADC_VBATT       3
+#define CHANNEL_ADC_VCOIN       4
+#define CHANNEL_ADC_VCHG        5
+#define CHANNEL_ADC_CHG_MONITOR 6
+#define CHANNEL_ADC_VPH_PWR     7
+#define CHANNEL_ADC_USB_VBUS    8
+#define CHANNEL_ADC_DIE_TEMP    9
+#define CHANNEL_ADC_DIE_TEMP_4K 0xa
+#define CHANNEL_ADC_XOTHERM     0xb
+#define CHANNEL_ADC_XOTHERM_4K  0xc
+#define CHANNEL_ADC_HDSET       0xd
+#define CHANNEL_ADC_MSM_THERM	0xe
+#define CHANNEL_ADC_625_REF	0xf
+#define CHANNEL_ADC_1250_REF	0x10
+#define CHANNEL_ADC_325_REF	0x11
+#define CHANNEL_ADC_FSM_THERM	0x12
+#define CHANNEL_ADC_PA_THERM	0x13
+
+enum {
+	CALIB_STARTED,
+	CALIB_NOT_REQUIRED = 0xffffffff,
+};
+
+struct linear_graph {
+	int32_t offset;
+	int32_t dy; /* Slope numerator */
+	int32_t dx; /* Slope denominator */
+};
+
+struct adc_map_pt {
+	int32_t x;
+	int32_t y;
+};
+
+struct adc_properties {
+	uint32_t adc_reference; /* milli-voltage for this adc */
+	uint32_t bitresolution;
+	bool bipolar;
+	uint32_t conversiontime;
+};
+
+struct chan_properties {
+	uint32_t gain_numerator;
+	uint32_t gain_denominator;
+	struct linear_graph *adc_graph;
+/* this maybe the same as adc_properties.ConversionTime
+   if channel does not change the adc properties */
+	uint32_t chan_conv_time;
+};
+
+struct msm_adc_channels {
+	char *name;
+	uint32_t channel_name;
+	uint32_t adc_dev_instance;
+	struct adc_access_fn *adc_access_fn;
+	uint32_t chan_path_type;
+	uint32_t adc_config_type;
+	uint32_t adc_calib_type;
+	int32_t (*chan_processor)(int32_t, const struct adc_properties *,
+		const struct chan_properties *, struct adc_chan_result *);
+
+};
+
+struct msm_adc_platform_data {
+	struct msm_adc_channels *channel;
+	uint32_t num_chan_supported;
+	uint32_t num_adc;
+	uint32_t chan_per_adc;
+	char **dev_names;
+	uint32_t target_hw;
+	uint32_t gpio_config;
+	u32 (*adc_gpio_enable) (int);
+	u32 (*adc_gpio_disable) (int);
+	u32 (*adc_fluid_enable) (void);
+	u32 (*adc_fluid_disable) (void);
+};
+
+enum hw_type {
+	MSM_7x30,
+	MSM_8x60,
+	FSM_9xxx,
+};
+
+enum epm_gpio_config {
+	MPROC_CONFIG,
+	APROC_CONFIG
+};
+
+enum adc_request {
+	START_OF_CONV,
+	END_OF_CONV,
+	START_OF_CALIBRATION,
+	END_OF_CALIBRATION,
+};
+
+struct adc_dev_spec {
+	uint32_t			hwmon_dev_idx;
+	struct dal_dev_spec {
+		uint32_t		dev_idx;
+		uint32_t		chan_idx;
+	} dal;
+};
+
+struct dal_conv_request {
+	struct dal_dev_spec		target;
+	void				*cb_h;
+};
+
+struct dal_adc_result {
+	uint32_t			status;
+	uint32_t			token;
+	uint32_t			dev_idx;
+	uint32_t			chan_idx;
+	int				physical;
+	uint32_t			percent;
+	uint32_t			microvolts;
+	uint32_t			reserved;
+};
+
+struct dal_conv_slot {
+	void				*cb_h;
+	struct dal_adc_result		result;
+	struct completion		comp;
+	struct list_head		list;
+	uint32_t			idx;
+	uint32_t			chan_idx;
+	bool				blocking;
+	struct msm_client_data		*client;
+};
+
+struct dal_translation {
+	uint32_t			dal_dev_idx;
+	uint32_t			hwmon_dev_idx;
+	uint32_t			hwmon_start;
+	uint32_t			hwmon_end;
+};
+
+struct msm_client_data {
+	struct list_head		complete_list;
+	bool				online;
+	int32_t				adc_chan;
+	uint32_t			num_complete;
+	uint32_t			num_outstanding;
+	wait_queue_head_t		data_wait;
+	wait_queue_head_t		outst_wait;
+	struct mutex lock;
+};
+
+struct adc_conv_slot {
+	void				*cb_h;
+	union {
+		struct adc_chan_result		result;
+		struct dal_adc_result		dal_result;
+	} conv;
+	struct completion		comp;
+	struct completion		*compk;
+	struct list_head		list;
+	uint32_t			idx;
+	enum adc_request		adc_request;
+	bool				blocking;
+	struct msm_client_data		*client;
+	struct work_struct		work;
+	struct chan_properties		chan_properties;
+	uint32_t			chan_path;
+	uint32_t			chan_adc_config;
+	uint32_t			chan_adc_calib;
+};
+
+struct adc_access_fn {
+	int32_t (*adc_select_chan_and_start_conv)(uint32_t,
+				struct adc_conv_slot*);
+	int32_t (*adc_read_adc_code)(uint32_t dev_instance, int32_t *data);
+	struct adc_properties *(*adc_get_properties)(uint32_t dev_instance);
+	void (*adc_slot_request)(uint32_t dev_instance,
+				struct adc_conv_slot **);
+	void (*adc_restore_slot)(uint32_t dev_instance,
+				struct adc_conv_slot *slot);
+	int32_t (*adc_calibrate)(uint32_t dev_instance, struct adc_conv_slot*,
+		int *);
+};
+
+void msm_adc_wq_work(struct work_struct *work);
+void msm_adc_conv_cb(void *context, u32 param, void *evt_buf, u32 len);
+int32_t adc_channel_open(uint32_t channel, void **h);
+int32_t adc_channel_close(void *h);
+int32_t adc_channel_request_conv(void *h, struct completion *conv_complete_evt);
+int32_t adc_channel_read_result(void *h, struct adc_chan_result *chan_result);
+int32_t adc_calib_request(void *h, struct completion *calib_complete_evt);
+#endif
+#endif /* __MSM_ADC_H */