/* * Author: Stan Gifford * Copyright (c) 2015 Intel Corporation. * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #include "adafruitss.h" #include #include using namespace upm; adafruitss::adafruitss(int bus,int i2c_address) { int n; int result; mraa_init(); m_i2c = mraa_i2c_init(bus); pca9685_addr = i2c_address; result=mraa_i2c_address(m_i2c, pca9685_addr); m_rx_tx_buf[0]=PCA9685_MODE1; m_rx_tx_buf[1]=0; result=mraa_i2c_write(m_i2c,m_rx_tx_buf,2); adafruitss::setPWMFreq(60); adafruitss::update(); } void adafruitss::setPWMFreq(float freq) { int result; freq *= 0.88; // Correct for overshoot in the frequency setting (see issue #11). float prescaleval = 25000000; prescaleval /= 4096; prescaleval /= freq; prescaleval -= 1; _pwm_frequency = 60.18; // FInal achieved frequency measured with Logic 8! _duration_1ms = ((4096*_pwm_frequency)/1000); // This is 1ms duration uint8_t prescale = floor(prescaleval + 0.5); result=mraa_i2c_address(m_i2c, pca9685_addr); uint8_t oldmode=0; oldmode = mraa_i2c_read_byte_data(m_i2c,PCA9685_MODE1); m_rx_tx_buf[0]=PCA9685_MODE1; m_rx_tx_buf[1]=0x10; // sleep result=mraa_i2c_address(m_i2c, pca9685_addr); result=mraa_i2c_write(m_i2c,m_rx_tx_buf,2); m_rx_tx_buf[0]=PCA9685_PRESCALE; m_rx_tx_buf[1]=prescale; result=mraa_i2c_address(m_i2c, pca9685_addr); result=mraa_i2c_write(m_i2c,m_rx_tx_buf,2); m_rx_tx_buf[0]=PCA9685_MODE1; m_rx_tx_buf[1]=0x00; result=mraa_i2c_address(m_i2c, pca9685_addr); result=mraa_i2c_write(m_i2c,m_rx_tx_buf,2); // result=mraa_i2c_write_byte_data(m_i2c,0x00,PCA9685_MODE1); usleep(5000); m_rx_tx_buf[0]=PCA9685_MODE1; m_rx_tx_buf[1]=0xa1; result=mraa_i2c_address(m_i2c, pca9685_addr); result=mraa_i2c_write(m_i2c,m_rx_tx_buf,2); } int adafruitss::update(void) { return MRAA_SUCCESS; } void adafruitss::servo(uint8_t port, uint8_t servo_type, uint16_t degrees) { // Set Servo values // Degrees is from 0 to 180 // servo_type: 0 = standard 1ms to 2ms // 1 = extended 0.6ms to 2.4ms // 2 = extended 0.8ms to 2.2ms uint16_t duration = 0; int result; int r2; if(degrees>180) degrees=180; // Ensure within bounds if (degrees<0) degrees=0; switch (servo_type) { case 0: // Standard Servo 1ms to 2ms duration = _duration_1ms + ((_duration_1ms*degrees)/180); break; case 1: // Extended Servo 0.6ms to 2.4ms, i.e. 1.8ms from 0 to 180 //duration = (_duration_1ms*0.6) + ((_duration_1ms*1.8*degrees)/180); simplified to.. duration = (_duration_1ms*0.6) + ((_duration_1ms*degrees)/100); break; case 2: // Extended Servo 0.8ms to 2.2ms, i.e. 1.4ms from 0 to 180 //duration = (_duration_1ms*0.8) + ((_duration_1ms*1.4*degrees)/180); simplified to.. duration = (_duration_1ms*0.8) + ((_duration_1ms*degrees)/128); break; case 3: // Extended Servo 0.9ms to 2.1ms, - GWS Mini STD BB servo //duration = (_duration_1ms*0.8) + ((_duration_1ms*1.4*degrees)/180); simplified to.. duration = (_duration_1ms*0.9) + ((_duration_1ms*degrees)/120); break; } result=mraa_i2c_address(m_i2c, pca9685_addr); m_rx_tx_buf[0]=LED0_REG+4*port; m_rx_tx_buf[1]=0; m_rx_tx_buf[2]=0; m_rx_tx_buf[3]=duration; m_rx_tx_buf[4]=duration>>8; result=mraa_i2c_write(m_i2c,m_rx_tx_buf,5); r2=result; }