| #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); |
| #ifdef CONFIG_SENSORS_MSM_ADC |
| 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); |
| #else |
| int32_t adc_channel_open(uint32_t channel, void **h) |
| { |
| pr_err("%s.not supported.\n", __func__); |
| return -ENODEV; |
| } |
| int32_t adc_channel_close(void *h) |
| { |
| pr_err("%s.not supported.\n", __func__); |
| return -ENODEV; |
| } |
| int32_t adc_channel_request_conv(void *h, struct completion *conv_complete_evt) |
| { |
| pr_err("%s.not supported.\n", __func__); |
| return -ENODEV; |
| } |
| int32_t adc_channel_read_result(void *h, struct adc_chan_result *chan_result) |
| { |
| pr_err("%s.not supported.\n", __func__); |
| return -ENODEV; |
| } |
| int32_t adc_calib_request(void *h, struct completion *calib_complete_evt) |
| { |
| pr_err("%s.not supported.\n", __func__); |
| return -ENODEV; |
| } |
| #endif /* CONFIG_SENSORS_MSM_ADC */ |
| #endif |
| #endif /* __MSM_ADC_H */ |