Mirror with some correction
Dependencies: mbed FastIO FastPWM USBDevice
Plunger/distanceSensor.h
- Committer:
- mjr
- Date:
- 2021-06-02
- Revision:
- 113:7330439f2ffc
- Parent:
- 111:42dc75fbe623
File content as of revision 113:7330439f2ffc:
// Plunger sensor type for distance sensors. // // This type of sensor measures the distance to a target by sending // optical or sound signals and watching for the reflection. There are // many types of these sensors, including sensors that measure the // intensity of reflected sound or light signals, sensors that measure // the round-trip time of "pings", and sensors that measure optical // parallax. // // The basic installation for this type of sensor involves placing the // sensor itself in a fixed location at one end of the plunger, pointing // down the length of the plunger, and placing a reflective target at // the end of the plunger. The target can simply be an ordinary plunger // tip, if the sensor is at the far end of the plunger facing forward // (facing the front of the cabinet). Alternatively, the target can // be a disk or similar object attached to the end of the plunger, and // the sensor can be placed at the front of the machine facing the target. // In either case, the sensor measures the distance to the target at any // given time, and we interpret that into the plunger position. // // Here are the specific sensor types we currently support: // // VCNL4010: An IR proximity sensor. This sensor shines an IR light at a // target and measures the intensity of the reflected light. This doesn't // measure distance per se, but since the intensity of a light source // falls off as the square of the distance, we can use the reflected // intensity as a proxy for the distance by calculating 1/sqrt(intensity). // The main reason to support this type of sensor is that it's used in the // VirtuaPin v3 plunger kit, and several people have requested support so // that they can move re-flash that kit using the Pinscape software and // continue using their existing plunger sensor. Many people might also // consider this sensor for new DIY builds, since it produces pretty good // results. It's not as accurate as a potentiometer or quadrature sensor, // but it yields low-noise results with good enough precision for smooth // on-screen animation (maybe around 1mm precision). Its main drawback // is that it's relatively slow (250 Hz maximum sampling rate), but it's // still fast enough to be usable. It has several virtues that might more // than orffset its technical limitations for many paople: it's easy to // set up physically, it's completely non-contact, and it's cheap (under // $10 for the Adafruit breakout board). // // VL6180X: This is an optical (IR) "time of flight" sensor that measures // the distance to the target by sending optical pings and timing the // return signal, converting the result to distance via the known speed // of light. This sensor has nominal 1mm precision, although its true // precision in testing is closer to 5mm. Sample times are around 16ms. // This makes the sensor acceptable but not great by Pinscape standards; // we generally consider 2.5ms read times and .25mm precision to be the // minimum standards. However, this sensor is inexpensive and easier to // set up than most of the better options, so it might be attractive to // some cab builders despite the quality tradeoffs. // // #ifndef _DISTANCESENSOR_H_ #define _DISTANCESENSOR_H_ #include "plunger.h" #include "VL6180X.h" #include "VCNL4010.h" // Base class for distance sensors class PlungerSensorDistance: public PlungerSensor { public: PlungerSensorDistance(int nativeScale) : PlungerSensor(nativeScale) { totalTime = 0; nRuns = 0; } // get the average scan time virtual uint32_t getAvgScanTime() { return uint32_t(totalTime / nRuns); } protected: // collect scan time statistics void collectScanTimeStats(uint32_t dt) { totalTime += dt; nRuns += 1; } // scan time statistics uint64_t totalTime; // total time consumed by all reads so far uint32_t nRuns; // number of runs so far }; // PlungerSensor interface implementation for VL6180X sensors. // // The VL6180X reports distances in millimeter quanta, so the native // sensor units are millimeters. A physical plunger has about 3" of // total travel, but leave a little extra padding for measurement // inaccuracies and other unusual situations, so'll use an actual // native scale of 150mm. class PlungerSensorVL6180X: public PlungerSensorDistance { public: PlungerSensorVL6180X(PinName sda, PinName scl, PinName gpio0) : PlungerSensorDistance(150), sensor(sda, scl, I2C_ADDRESS, gpio0, true) { } // fixed I2C bus address for the VL6180X static const int I2C_ADDRESS = 0x29; virtual void init() { // initialize the sensor and set the default configuration sensor.init(); sensor.setDefaults(); // start a reading sensor.startRangeReading(); } virtual bool ready() { // make sure a reading has been initiated sensor.startRangeReading(); // check if a reading is ready return sensor.rangeReady(); } virtual bool readRaw(PlungerReading &r) { // if we have a new reading ready, collect it if (sensor.rangeReady()) { // Get the range reading. Note that we already know that the // sensor has a reading ready, so it shouldn't be possible to // time out on the read. (The sensor could have timed out on // convergence, but if it did, that's in the past already so // it's not something we have to wait for now.) uint8_t d; uint32_t t, dt; lastErr = sensor.getRange(d, t, dt, 100); // if we got a reading, update the last reading if (lastErr == 0) { // save the new reading last.pos = d; last.t = t; // collect scan time statistics collectScanTimeStats(dt); } // start a new reading sensor.startRangeReading(); } // return the most recent reading r = last; return lastErr == 0; } protected: // underlying sensor interface VL6180X sensor; // last reading and error status PlungerReading last; int lastErr; }; // PlungerSensor interface implementation for VCNL4010 IR proximity sensors // // Our hardware interface for this sensor reports distances in abstract // units that fit a 16-bit int, so the native distance scale is 0..65535. // (The sensor itself doesn't have a native distance scale per se, since // it reports results in terms of the intensity of the reflected light. // This is related to the distance by an inverse square law, so since we // have to do some math on the raw readings anyway to convert them to // distances, we can choose whatever units we want for the conversion. // We choose units that are convenient for our purposes at the joystick // layer, given the 16-bit field we use to report the position back to // the PC.) // // The iredCurrent parameter sets the brightness of the sensor's IR LED, // which serves as the light source for the reflected light intensity // readings used for proximity measurements. This is given in units of // 10mA, so 1 means 10mA, 2 means 20mA, etc. Valid values are from 1 // (10mA) to 20 (200mA). // class PlungerSensorVCNL4010: public PlungerSensorDistance { public: PlungerSensorVCNL4010(PinName sda, PinName scl, int iredCurrent) : PlungerSensorDistance(65535), sensor(sda, scl, true, iredCurrent) { } virtual void init() { // initialize the sensor sensor.init(); // start a reading sensor.startProxReading(); } virtual bool ready() { // check if a reading is ready return sensor.proxReady(); } virtual bool readRaw(PlungerReading &r) { // if we have a new reading ready, collect it if (sensor.proxReady()) { // Get the proximity count reading. Note that we already know // that the sensor has a reading ready, so it shouldn't be // possible to time out on the read. int rawCount; uint32_t t, dt; lastErr = sensor.getProx(rawCount, t, dt, 100); // if we got a reading, update the last reading if (lastErr == 0) { // run the proximity count through the jitter filter int filteredCount = jitterFilter(rawCount); // convert the count to a distance, using the filtered count int dist = sensor.countToDistance(filteredCount); // save the new reading last.pos = dist; last.t = t; lastFilteredCount = filteredCount; lastRawCount = rawCount; // collect scan time statistics collectScanTimeStats(dt); } } // return the most recent reading r = last; return lastErr == 0; } // The VCNL4010 applies jitter filtering to the physical sensor reading // instead of to the distance reading. This produces much better results // for this sensor because the sensor's distance resolution gets lower // at longer distances, so the conversion to distance tends to amplify // noise quite a bit at the distant end. It's therefore important to // do the noise reduction in the brightness domain, before that // amplification takes place. virtual int postJitterFilter(int pos) { return pos; } // Send a status report for the config tool sensor viewer virtual void sendStatusReport(class USBJoystick &js, uint8_t flags) { // send the common status report PlungerSensor::sendStatusReport(js, flags); // send the extra VCNL4010 sensor status report js.sendPlungerStatusVCNL4010(lastFilteredCount, lastRawCount); } // Restore the saved calibration data from the configuration. The // main loop calls this at initialization time to pass us saved // private configuration data. The VCNL4010 uses this to store the // minimum proximity count reading observed during calibration, which // it uses to figure the scaling factor for the 1/sqrt(intensity) // distance calculation. virtual void restoreCalibration(Config &config) { // restore the saved minimum count reading sensor.restoreCalibration(config); } // Begin calibration. The main loop calls this when the user // initiates a calibration cycle. The VCNL4010 code uses this to // reset its internal record of the proximity minimum. virtual void beginCalibration(Config &) { sensor.beginCalibration(); } // End calibration. The main loop calls this when a calibration // cycle finishes. The VCNL4010 code uses this to save the minimum // count value observed during the calibration interval, and to // calculate the new scaling factor for the 1/sqrt(intensity) // distance calculation. virtual void endCalibration(Config &config) { // let the sensor figure the new scaling factor sensor.endCalibration(config); } protected: // underlying sensor interface VCNL4010 sensor; // last reading and error status PlungerReading last; int lastFilteredCount; int lastRawCount; int lastErr; }; #endif