Hi David, On Fri, Jul 13, 2018 at 01:42:35PM -0700, David Frey wrote: > Hi Himanshu Jha, > > First a bit of background. I'm working on a device which will contain a > bme680 sensor. A colleague of mine started work on a Linux kernel driver > for the chip a little while ago. The (WIP) driver can be found here: > https://github.com/mangOH/mangOH/tree/master/linux_kernel_modules/bme680 Great! > This driver is written targeting an older kernel (3.18.x) because that's the > kernel we're stuck on for now. Rather than writing the driver from scratch, > what we did was write the Linux kernel driver as a wrapper around the Bosch > code. My theory at the time was that Bosch made the chip, so they probably > know what they're doing when it comes to writing a driver library. After > having looked at the code in more detail, I'm less confident that our > approach was the best one. I'm not attempting to upstream the driver built > by my colleague and I'm not trying to request review of this code either. I > simply want to make you aware of it so that you can look at it to get some > ideas. Thanks for taking your time to review. > I have included a number of comments on your driver below. Keep up the good > work! > > >+++ b/drivers/iio/chemical/bme680.h > >@@ -0,0 +1,99 @@ > >+/* SPDX-License-Identifier: GPL-2.0 */ > >+#ifndef BME680_H_ > >+#define BME680_H_ > >+ > >+#define BME680_REG_CHIP_I2C_ID 0xD0 > >+#define BME680_REG_CHIP_SPI_ID 0x50 > >+#define BME680_CHIP_ID_VAL 0x61 > Try to be consistent with the indenting of the defines. I think this would > be clearest: > #define BME680_REG_X 0x00 > #define BME680_X_FOO_EN_MASK BIT(0) > #define BME680_X_BAR_MASK GENMASK(3, 1) > #define BME680_BAR_VAL1 3 > #define BME680_BAR_VAL2 7 > > This way the register, field definition and field values are all visually > distinctive. I have used this pattern everywhere where applicable. But not applied for *_VAL, would definitely follow this up. > >+#define BME680_REG_SOFT_RESET 0xE0 > The datasheet says that the soft reset register differs for I2C and SPI. > For I2C it is 0xE0 and for SPI it is 0x60 when page 0 is selected. That's really a stupid mistake :( I have exported these individual initialization code in the I2C & SPI drivers respectively but it slipped my mind somehow. This device has peculiar characteristics in register addressing. I will correct this in next version. > >+#define BME680_CMD_SOFTRESET 0xB6 > >+#define BME680_REG_STATUS 0x73 > >+#define BME680_SPI_MEM_PAGE_BIT BIT(4) > >+#define BME680_SPI_MEM_PAGE_1_VAL 1 > >+ > >+#define BME680_OSRS_TEMP_X(osrs_t) ((osrs_t) << 5) > >+#define BME680_OSRS_PRESS_X(osrs_p) ((osrs_p) << 2) > >+#define BME680_OSRS_HUMID_X(osrs_h) ((osrs_h) << 0) > You could use the FIELD_PREP macro from <linux/bitfield.h> to eliminate the > need for these macros. For example: > ctrl_meas_reg = FIELD_PREP(BME680_OSRS_TEMP_MASK, temp_val) | > FIELD_PREP(BME680_OSRS_PRESS_MASK, press_val) | > FIELD_PREP(BME880_MODE_MASK, mode_val); Ah, yes! I didn't knew about these magic macros. It will remove some log2() computation hacks from my code. > >+ > >+#define BME680_REG_TEMP_MSB 0x22 > >+#define BME680_REG_PRESS_MSB 0x1F > >+#define BM6880_REG_HUMIDITY_MSB 0x25 > >+#define BME680_REG_GAS_MSB 0x2A > >+#define BME680_REG_GAS_R_LSB 0x2B > >+#define BME680_GAS_STAB_BIT BIT(4) > >+ > >+#define BME680_REG_CTRL_HUMIDITY 0x72 > >+#define BME680_OSRS_HUMIDITY_MASK GENMASK(2, 0) > >+ > >+#define BME680_REG_CTRL_MEAS 0x74 > >+#define BME680_OSRS_TEMP_MASK GENMASK(7, 5) > >+#define BME680_OSRS_PRESS_MASK GENMASK(4, 2) > >+#define BME680_MODE_MASK GENMASK(1, 0) > >+ > >+#define BME680_MODE_FORCED BIT(0) > >+#define BME680_MODE_SLEEP 0 > This should be: > #define BME680_MODE_SLEEP 0 > #define BME680_MODE_FORCED 1 Yes, this is much clearer and removes ambiguity. > >+/* Taken from Bosch BME680 API */ > > I think there should be a link to the Bosch code > (https://github.com/BoschSensortec/BME680_driver/) somewhere within the > comments of this file. Maybe it belongs at the top of this file? I planned to add: https://github.com/BoschSensortec/BME680_driver/blob/63bb5336db4659519860832be2738c685133aa33/bme680.c#L876 to here and likewise to other compensate functions. But these links may change(if somehow they plan to migrate to Gitlab), long lines are not welcomed. You could also notice that I haven't included datasheet link at the top of this file. Well, most of the companies change the links when releasing the new Revision(Rev. A,B,...) so it is likely that the link would be dead/old lying at the top of source. Therefore, I added both links to Bosch API and datasheet only in the commmit log(also suggested by my mentor). > >+static s32 bme680_compensate_temp(struct bme680_data *data, > >+ s32 adc_temp) > >+{ > >+ struct bme680_calib *calib = &data->bme680; > >+ s64 var1, var2, var3, calc_temp; > >+ > >+ var1 = ((s32) adc_temp >> 3) - ((s32) calib->par_t1 << 1); > >+ var2 = (var1 * (s32) calib->par_t2) >> 11; > >+ var3 = ((var1 >> 1) * (var1 >> 1)) >> 12; > >+ var3 = ((var3) * ((s32) calib->par_t3 << 4)) >> 14; > >+ data->t_fine = (s32) (var2 + var3); > >+ calc_temp = (s16) (((data->t_fine * 5) + 128) >> 8); > >+ > >+ return calc_temp; > I think the meaning of the s32 returned should be documented. Based on code > elsewhere in the driver I'm guessing it's degrees celcius * 100. The same > comment applies to the other compensate functions. Yes, this function returns values like 3253(32.53degC), which I divide in the read_temp() at last few lines: *val = bme680_compensate_temp(); *val2 = 100; return IIO_VAL_FRACTIONAL; You can only document when you have "official" references, I mean these *should* have been documented in datasheet. And then we could say: "Temperature has 100 degC .... Pg 21" But these resolution information is what I got from Bosch when I contacted them and also evident from README.md https://github.com/BoschSensortec/BME680_driver/blob/63bb5336db4659519860832be2738c685133aa33/README.md#example-for-reading-all-sensor-data "printf("T: %.2f degC, P: %.2f hPa, H %.2f %%rH ", data.temperature / 100.0f, data.pressure / 100.0f, data.humidity / 1000.0f );" And I am pretty sure they will add these information in the coming revision of datasheet. I have looked at the various family of sensors from Bosch and they even mention the optimal oversampling ratio, filter coefficient, etc., for say "using in weather stations"(Take a look at bme280 datasheet) [ market policy is what I heard somewhere ;) ] > >+static int bme680_chip_config(struct bme680_data *data) > >+{ > >+ struct device *dev = regmap_get_device(data->regmap); > >+ int ret; > >+ u8 osrs = BME680_OSRS_HUMID_X(data->oversampling_humid + 1); > >+ /* > >+ * Highly recommended to set oversampling of humidity before > >+ * temperature/pressure oversampling. > >+ */ > I think you are referring to this snippet from the datasheet: "It is highly > recommended to set first osrs_h<2:0> followed by osrs_t<2:0> and osrs_p<2:0> > in one write command (see Section 3.3)." My interpretation of this is that > they are saying that you should do one bulk write rather than writing the > fields individually. Maybe they're just recommending this for efficiency > reasons. I'm not really sure though. I read through section 3.3 and I > couldn't find any justification to backup this rather prominent suggestion. Yes, I referred to the same segment of article from datasheet. But how do you do a bulk_write() for *two different* registers ? We have humidity bit(osrs_h<2:0) at Ctrl_hum(0x72) register and temperature(osrs_t), pressure bit(osrs_p) at Ctrl_meas(0x74) register. So, I think it was setting humidity first perhaps and then doing the rest. Even Bosch is doing the reverse of what they say in the datasheet: https://github.com/BoschSensortec/BME680_driver/blob/63bb5336db4659519860832be2738c685133aa33/bme680.c#L464 They are setting Temperature & Pressure _before_ humidity oversampling setting. Weird!? > >+static int bme680_read_temp(struct bme680_data *data, > >+ int *val, int *val2) > >+{ > >+ struct device *dev = regmap_get_device(data->regmap); > >+ int ret = 0; > >+ __be32 tmp = 0; > >+ s32 adc_temp, comp_temp; > >+ > >+ /* set forced mode to trigger measurement */ > >+ ret = bme680_set_mode(data, true); > >+ if (ret < 0) > >+ return ret; > >+ > >+ ret = regmap_bulk_read(data->regmap, BME680_REG_TEMP_MSB, > >+ (u8 *) &tmp, 3); > >+ if (ret < 0) { > >+ dev_err(dev, "failed to read temperature\n"); > >+ return ret; > >+ } > I think the value you're reading from the register may actually be from a > previous request. I think you need to poll meas_status_0 (0x1D) field > new_data_0 (bit 7) to wait for new data after setting the mode to forced. > You can see that's what Bosch's code does: > https://github.com/BoschSensortec/BME680_driver/blob/63bb5336db4659519860832be2738c685133aa33/bme680.c#L1227 Yes, we can do that but, for a while, I am following my GSoC proposal timeline. And I have planned these check_bits function later, but since this check_bits function was imperative in gas sensing, therefore I included them now. My plan in incremental changes and this patch is kindof minimal. I have tested the sensor several times and found no errors in readings for T P H G readings so far. The problem arises that this sensor is made to work in a T->P->H->G fashion and every channel is mostly dependent on the other. And IIO driver design pattern isn't the best choice it. You can't just take a single channel readings by running Bosch Code because it is not designed like that. For instance: we need t_fine values for pressure/humidity compensation functions which we only get when reading temperature. So, you need to run the temperaure cycle if you need pressure/humidity values. And this is what I did by doing a dummy read_temp(data, NULL, NULL) to get the t_fine value. I had also planned to add heater_profile_duration: https://github.com/BoschSensortec/BME680_driver/blob/63bb5336db4659519860832be2738c685133aa33/bme680.c#L647 But what if I only read gas channel ? Then this function needs to be tweaked accordingly. But without these functions my code works fine, but I will surely add them later. There are also function that can be added for eg. to add a delay during startup: Start-up time -- tstartup -- Time to first communication -- 2ms This delay should be added after power-on reset. I am currently considering only the core functionality for now, later incrementally add these function. I am currenlty supplying default values for heater temperature & duration at probe and planned to add syfs attributes for user to specify these values. But somehow it got complicated and my mentor advised to just to the needful now and later we can add various support interfaces. We can also add interface for turning on/off optional IIR filter. The another problem also arise is that each you want a reading, you need to set the FORCED mode. But Bosch Code doesn't support individual channel reading as we do in IIO, they don;t need to care about this information. But on other side in IIO, if you want pressure reading you do: # cat in_pressure_humidityrelative .. and then behind the scenes FORCED mode is set and you get the reading. But then again, if you want temp & pressure both: # cat in_temp_input in_pressure_humidityrelative .. and here we set FORCED mode *twice* for every measurement. And then datasheet say: At Page 14/50 ------------ 3. Sensor usage 3.1 Sensor modes "It is important to note that, further mode change commands or other write commands to the control registers are ignored until the mode change command has been executed. All control registers should be set to the desired values before writing to the mode register." So, we need to take care of these stuff too! And then there are a lot of things to discuss.... Thanks you so much for the feedback, David! :) And if at some point I have said something stupid, then please forgive me. I am a 3rd year undergrad student and started with IIO few months back, and not a Bosch driver developer ;) -- Himanshu Jha Undergraduate Student Department of Electronics & Communication Guru Tegh Bahadur Institute of Technology -- To unsubscribe from this list: send the line "unsubscribe linux-iio" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html