initial commit
This commit is contained in:
@@ -0,0 +1,194 @@
|
||||
/***************************************************
|
||||
This is a library for our Adafruit 16-channel PWM & Servo driver
|
||||
|
||||
Pick one up today in the adafruit shop!
|
||||
------> http://www.adafruit.com/products/815
|
||||
|
||||
These displays use I2C to communicate, 2 pins are required to
|
||||
interface.
|
||||
|
||||
Adafruit invests time and resources providing this open source code,
|
||||
please support Adafruit and open-source hardware by purchasing
|
||||
products from Adafruit!
|
||||
|
||||
Written by Limor Fried/Ladyada for Adafruit Industries.
|
||||
BSD license, all text above must be included in any redistribution
|
||||
****************************************************/
|
||||
|
||||
#include <Adafruit_PWMServoDriver.h>
|
||||
#include <Wire.h>
|
||||
|
||||
// Set to true to print some debug messages, or false to disable them.
|
||||
//#define ENABLE_DEBUG_OUTPUT
|
||||
|
||||
|
||||
/**************************************************************************/
|
||||
/*!
|
||||
@brief Instantiates a new PCA9685 PWM driver chip with the I2C address on the Wire interface. On Due we use Wire1 since its the interface on the 'default' I2C pins.
|
||||
@param addr The 7-bit I2C address to locate this chip, default is 0x40
|
||||
*/
|
||||
/**************************************************************************/
|
||||
Adafruit_PWMServoDriver::Adafruit_PWMServoDriver(uint8_t addr) {
|
||||
_i2caddr = addr;
|
||||
|
||||
#if defined(ARDUINO_SAM_DUE)
|
||||
_i2c = &Wire1;
|
||||
#else
|
||||
_i2c = &Wire;
|
||||
#endif
|
||||
}
|
||||
|
||||
/**************************************************************************/
|
||||
/*!
|
||||
@brief Instantiates a new PCA9685 PWM driver chip with the I2C address on a TwoWire interface
|
||||
@param i2c A pointer to a 'Wire' compatible object that we'll use to communicate with
|
||||
@param addr The 7-bit I2C address to locate this chip, default is 0x40
|
||||
*/
|
||||
/**************************************************************************/
|
||||
Adafruit_PWMServoDriver::Adafruit_PWMServoDriver(TwoWire *i2c, uint8_t addr) {
|
||||
_i2c = i2c;
|
||||
_i2caddr = addr;
|
||||
}
|
||||
|
||||
/**************************************************************************/
|
||||
/*!
|
||||
@brief Setups the I2C interface and hardware
|
||||
*/
|
||||
/**************************************************************************/
|
||||
void Adafruit_PWMServoDriver::begin(void) {
|
||||
_i2c->begin();
|
||||
reset();
|
||||
// set a default frequency
|
||||
setPWMFreq(1000);
|
||||
}
|
||||
|
||||
|
||||
/**************************************************************************/
|
||||
/*!
|
||||
@brief Sends a reset command to the PCA9685 chip over I2C
|
||||
*/
|
||||
/**************************************************************************/
|
||||
void Adafruit_PWMServoDriver::reset(void) {
|
||||
write8(PCA9685_MODE1, 0x80);
|
||||
delay(10);
|
||||
}
|
||||
|
||||
/**************************************************************************/
|
||||
/*!
|
||||
@brief Sets the PWM frequency for the entire chip, up to ~1.6 KHz
|
||||
@param freq Floating point frequency that we will attempt to match
|
||||
*/
|
||||
/**************************************************************************/
|
||||
void Adafruit_PWMServoDriver::setPWMFreq(float freq) {
|
||||
#ifdef ENABLE_DEBUG_OUTPUT
|
||||
Serial.print("Attempting to set freq ");
|
||||
Serial.println(freq);
|
||||
#endif
|
||||
|
||||
freq *= 0.9; // Correct for overshoot in the frequency setting (see issue #11).
|
||||
float prescaleval = 25000000;
|
||||
prescaleval /= 4096;
|
||||
prescaleval /= freq;
|
||||
prescaleval -= 1;
|
||||
|
||||
#ifdef ENABLE_DEBUG_OUTPUT
|
||||
Serial.print("Estimated pre-scale: "); Serial.println(prescaleval);
|
||||
#endif
|
||||
|
||||
uint8_t prescale = floor(prescaleval + 0.5);
|
||||
#ifdef ENABLE_DEBUG_OUTPUT
|
||||
Serial.print("Final pre-scale: "); Serial.println(prescale);
|
||||
#endif
|
||||
|
||||
uint8_t oldmode = read8(PCA9685_MODE1);
|
||||
uint8_t newmode = (oldmode&0x7F) | 0x10; // sleep
|
||||
write8(PCA9685_MODE1, newmode); // go to sleep
|
||||
write8(PCA9685_PRESCALE, prescale); // set the prescaler
|
||||
write8(PCA9685_MODE1, oldmode);
|
||||
delay(5);
|
||||
write8(PCA9685_MODE1, oldmode | 0xa0); // This sets the MODE1 register to turn on auto increment.
|
||||
|
||||
#ifdef ENABLE_DEBUG_OUTPUT
|
||||
Serial.print("Mode now 0x"); Serial.println(read8(PCA9685_MODE1), HEX);
|
||||
#endif
|
||||
}
|
||||
|
||||
/**************************************************************************/
|
||||
/*!
|
||||
@brief Sets the PWM output of one of the PCA9685 pins
|
||||
@param num One of the PWM output pins, from 0 to 15
|
||||
@param on At what point in the 4096-part cycle to turn the PWM output ON
|
||||
@param off At what point in the 4096-part cycle to turn the PWM output OFF
|
||||
*/
|
||||
/**************************************************************************/
|
||||
void Adafruit_PWMServoDriver::setPWM(uint8_t num, uint16_t on, uint16_t off) {
|
||||
#ifdef ENABLE_DEBUG_OUTPUT
|
||||
Serial.print("Setting PWM "); Serial.print(num); Serial.print(": "); Serial.print(on); Serial.print("->"); Serial.println(off);
|
||||
#endif
|
||||
|
||||
_i2c->beginTransmission(_i2caddr);
|
||||
_i2c->write(LED0_ON_L+4*num);
|
||||
_i2c->write(on);
|
||||
_i2c->write(on>>8);
|
||||
_i2c->write(off);
|
||||
_i2c->write(off>>8);
|
||||
_i2c->endTransmission();
|
||||
}
|
||||
|
||||
/**************************************************************************/
|
||||
/*!
|
||||
@brief Helper to set pin PWM output. Sets pin without having to deal with on/off tick placement and properly handles a zero value as completely off and 4095 as completely on. Optional invert parameter supports inverting the pulse for sinking to ground.
|
||||
@param num One of the PWM output pins, from 0 to 15
|
||||
@param val The number of ticks out of 4096 to be active, should be a value from 0 to 4095 inclusive.
|
||||
@param invert If true, inverts the output, defaults to 'false'
|
||||
*/
|
||||
/**************************************************************************/
|
||||
void Adafruit_PWMServoDriver::setPin(uint8_t num, uint16_t val, bool invert)
|
||||
{
|
||||
// Clamp value between 0 and 4095 inclusive.
|
||||
val = min(val, (uint16_t)4095);
|
||||
if (invert) {
|
||||
if (val == 0) {
|
||||
// Special value for signal fully on.
|
||||
setPWM(num, 4096, 0);
|
||||
}
|
||||
else if (val == 4095) {
|
||||
// Special value for signal fully off.
|
||||
setPWM(num, 0, 4096);
|
||||
}
|
||||
else {
|
||||
setPWM(num, 0, 4095-val);
|
||||
}
|
||||
}
|
||||
else {
|
||||
if (val == 4095) {
|
||||
// Special value for signal fully on.
|
||||
setPWM(num, 4096, 0);
|
||||
}
|
||||
else if (val == 0) {
|
||||
// Special value for signal fully off.
|
||||
setPWM(num, 0, 4096);
|
||||
}
|
||||
else {
|
||||
setPWM(num, 0, val);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*******************************************************************************************/
|
||||
|
||||
uint8_t Adafruit_PWMServoDriver::read8(uint8_t addr) {
|
||||
_i2c->beginTransmission(_i2caddr);
|
||||
_i2c->write(addr);
|
||||
_i2c->endTransmission();
|
||||
|
||||
_i2c->requestFrom((uint8_t)_i2caddr, (uint8_t)1);
|
||||
return _i2c->read();
|
||||
}
|
||||
|
||||
void Adafruit_PWMServoDriver::write8(uint8_t addr, uint8_t d) {
|
||||
_i2c->beginTransmission(_i2caddr);
|
||||
_i2c->write(addr);
|
||||
_i2c->write(d);
|
||||
_i2c->endTransmission();
|
||||
}
|
||||
@@ -0,0 +1,69 @@
|
||||
/***************************************************
|
||||
This is a library for our Adafruit 16-channel PWM & Servo driver
|
||||
|
||||
Pick one up today in the adafruit shop!
|
||||
------> http://www.adafruit.com/products/815
|
||||
|
||||
These displays use I2C to communicate, 2 pins are required to
|
||||
interface. For Arduino UNOs, thats SCL -> Analog 5, SDA -> Analog 4
|
||||
|
||||
Adafruit invests time and resources providing this open source code,
|
||||
please support Adafruit and open-source hardware by purchasing
|
||||
products from Adafruit!
|
||||
|
||||
Written by Limor Fried/Ladyada for Adafruit Industries.
|
||||
BSD license, all text above must be included in any redistribution
|
||||
****************************************************/
|
||||
|
||||
#ifndef _ADAFRUIT_PWMServoDriver_H
|
||||
#define _ADAFRUIT_PWMServoDriver_H
|
||||
|
||||
#if ARDUINO >= 100
|
||||
#include "Arduino.h"
|
||||
#else
|
||||
#include "WProgram.h"
|
||||
#endif
|
||||
#include "Wire.h"
|
||||
|
||||
#define PCA9685_SUBADR1 0x2
|
||||
#define PCA9685_SUBADR2 0x3
|
||||
#define PCA9685_SUBADR3 0x4
|
||||
|
||||
#define PCA9685_MODE1 0x0
|
||||
#define PCA9685_PRESCALE 0xFE
|
||||
|
||||
#define LED0_ON_L 0x6
|
||||
#define LED0_ON_H 0x7
|
||||
#define LED0_OFF_L 0x8
|
||||
#define LED0_OFF_H 0x9
|
||||
|
||||
#define ALLLED_ON_L 0xFA
|
||||
#define ALLLED_ON_H 0xFB
|
||||
#define ALLLED_OFF_L 0xFC
|
||||
#define ALLLED_OFF_H 0xFD
|
||||
|
||||
/**************************************************************************/
|
||||
/*!
|
||||
@brief Class that stores state and functions for interacting with PCA9685 PWM chip
|
||||
*/
|
||||
/**************************************************************************/
|
||||
class Adafruit_PWMServoDriver {
|
||||
public:
|
||||
Adafruit_PWMServoDriver(uint8_t addr = 0x40);
|
||||
Adafruit_PWMServoDriver(TwoWire *I2C, uint8_t addr = 0x40);
|
||||
void begin(void);
|
||||
void reset(void);
|
||||
void setPWMFreq(float freq);
|
||||
void setPWM(uint8_t num, uint16_t on, uint16_t off);
|
||||
void setPin(uint8_t num, uint16_t val, bool invert=false);
|
||||
|
||||
private:
|
||||
uint8_t _i2caddr;
|
||||
|
||||
TwoWire *_i2c;
|
||||
|
||||
uint8_t read8(uint8_t addr);
|
||||
void write8(uint8_t addr, uint8_t d);
|
||||
};
|
||||
|
||||
#endif
|
||||
50
libraries/Adafruit_PWM_Servo_Driver_Library/README.md
Normal file
50
libraries/Adafruit_PWM_Servo_Driver_Library/README.md
Normal file
@@ -0,0 +1,50 @@
|
||||
# Adafruit PCA9685 PWM Servo Driver Library [](https://travis-ci.org/adafruit/Adafruit_PWMServoDriver)
|
||||
|
||||
This is a library for our Adafruit 16-channel PWM & Servo driver, shield or FeatherWing
|
||||
|
||||
<img src="https://cdn-shop.adafruit.com/970x728/815-04.jpg" height="300"/>
|
||||
|
||||
Pick one up today in the adafruit shop!
|
||||
* https://www.adafruit.com/products/815
|
||||
* https://www.adafruit.com/product/1411
|
||||
* https://www.adafruit.com/product/2928
|
||||
|
||||
These drivers use I2C to communicate, 2 pins are required to interface.
|
||||
|
||||
Adafruit invests time and resources providing this open source code, please support Adafruit and open-source hardware by purchasing products from Adafruit!
|
||||
|
||||
Written by Limor Fried/Ladyada for Adafruit Industries. BSD license, check license.txt for more information.
|
||||
|
||||
All text above must be included in any redistribution
|
||||
|
||||
<!-- START COMPATIBILITY TABLE -->
|
||||
|
||||
## Compatibility
|
||||
|
||||
MCU | Tested Works | Doesn't Work | Not Tested | Notes
|
||||
------------------ | :----------: | :----------: | :---------: | -----
|
||||
Atmega328 @ 16MHz | X | | |
|
||||
Atmega328 @ 12MHz | X | | |
|
||||
Atmega32u4 @ 16MHz | X | | |
|
||||
Atmega32u4 @ 8MHz | X | | |
|
||||
ESP8266 | X | | |
|
||||
Atmega2560 @ 16MHz | X | | |
|
||||
ATSAM3X8E | X | | | Use SDA1/SCL1
|
||||
ATSAM21D | X | | |
|
||||
ATtiny85 @ 16MHz | X | | |
|
||||
ATtiny85 @ 8MHz | X | | |
|
||||
Intel Curie @ 32MHz | | | X |
|
||||
STM32F2 | | | X |
|
||||
|
||||
* ATmega328 @ 16MHz : Arduino UNO, Adafruit Pro Trinket 5V, Adafruit Metro 328, Adafruit Metro Mini
|
||||
* ATmega328 @ 12MHz : Adafruit Pro Trinket 3V
|
||||
* ATmega32u4 @ 16MHz : Arduino Leonardo, Arduino Micro, Arduino Yun, Teensy 2.0
|
||||
* ATmega32u4 @ 8MHz : Adafruit Flora, Bluefruit Micro
|
||||
* ESP8266 : Adafruit Huzzah
|
||||
* ATmega2560 @ 16MHz : Arduino Mega
|
||||
* ATSAM3X8E : Arduino Due
|
||||
* ATSAM21D : Arduino Zero, M0 Pro
|
||||
* ATtiny85 @ 16MHz : Adafruit Trinket 5V
|
||||
* ATtiny85 @ 8MHz : Adafruit Gemma, Arduino Gemma, Adafruit Trinket 3V
|
||||
|
||||
<!-- END COMPATIBILITY TABLE -->
|
||||
@@ -0,0 +1,49 @@
|
||||
/***************************************************
|
||||
This is an example for our Adafruit 16-channel PWM & Servo driver
|
||||
GPIO test - this will set a pin high/low
|
||||
|
||||
Pick one up today in the adafruit shop!
|
||||
------> http://www.adafruit.com/products/815
|
||||
|
||||
These drivers use I2C to communicate, 2 pins are required to
|
||||
interface.
|
||||
|
||||
Adafruit invests time and resources providing this open source code,
|
||||
please support Adafruit and open-source hardware by purchasing
|
||||
products from Adafruit!
|
||||
|
||||
Written by Limor Fried/Ladyada for Adafruit Industries.
|
||||
BSD license, all text above must be included in any redistribution
|
||||
****************************************************/
|
||||
|
||||
#include <Wire.h>
|
||||
#include <Adafruit_PWMServoDriver.h>
|
||||
|
||||
// called this way, it uses the default address 0x40
|
||||
Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver();
|
||||
// you can also call it with a different address you want
|
||||
//Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver(0x41);
|
||||
// you can also call it with a different address and I2C interface
|
||||
//Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver(&Wire, 0x40);
|
||||
|
||||
void setup() {
|
||||
Serial.begin(9600);
|
||||
Serial.println("GPIO test!");
|
||||
|
||||
pwm.begin();
|
||||
pwm.setPWMFreq(1000); // Set to whatever you like, we don't use it in this demo!
|
||||
|
||||
// if you want to really speed stuff up, you can go into 'fast 400khz I2C' mode
|
||||
// some i2c devices dont like this so much so if you're sharing the bus, watch
|
||||
// out for this!
|
||||
Wire.setClock(400000);
|
||||
}
|
||||
|
||||
void loop() {
|
||||
// Drive each pin in a 'wave'
|
||||
for (uint8_t pin=0; pin<16; pin++) {
|
||||
pwm.setPWM(pin, 4096, 0); // turns pin fully on
|
||||
delay(100);
|
||||
pwm.setPWM(pin, 0, 4096); // turns pin fully off
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,52 @@
|
||||
/***************************************************
|
||||
This is an example for our Adafruit 16-channel PWM & Servo driver
|
||||
PWM test - this will drive 16 PWMs in a 'wave'
|
||||
|
||||
Pick one up today in the adafruit shop!
|
||||
------> http://www.adafruit.com/products/815
|
||||
|
||||
These drivers use I2C to communicate, 2 pins are required to
|
||||
interface.
|
||||
|
||||
Adafruit invests time and resources providing this open source code,
|
||||
please support Adafruit and open-source hardware by purchasing
|
||||
products from Adafruit!
|
||||
|
||||
Written by Limor Fried/Ladyada for Adafruit Industries.
|
||||
BSD license, all text above must be included in any redistribution
|
||||
****************************************************/
|
||||
|
||||
#include <Wire.h>
|
||||
#include <Adafruit_PWMServoDriver.h>
|
||||
|
||||
// called this way, it uses the default address 0x40
|
||||
Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver();
|
||||
// you can also call it with a different address you want
|
||||
//Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver(0x41);
|
||||
// you can also call it with a different address and I2C interface
|
||||
//Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver(&Wire, 0x40);
|
||||
|
||||
void setup() {
|
||||
Serial.begin(9600);
|
||||
Serial.println("16 channel PWM test!");
|
||||
|
||||
pwm.begin();
|
||||
pwm.setPWMFreq(1600); // This is the maximum PWM frequency
|
||||
|
||||
// if you want to really speed stuff up, you can go into 'fast 400khz I2C' mode
|
||||
// some i2c devices dont like this so much so if you're sharing the bus, watch
|
||||
// out for this!
|
||||
Wire.setClock(400000);
|
||||
}
|
||||
|
||||
void loop() {
|
||||
// Drive each PWM in a 'wave'
|
||||
for (uint16_t i=0; i<4096; i += 8) {
|
||||
for (uint8_t pwmnum=0; pwmnum < 16; pwmnum++) {
|
||||
pwm.setPWM(pwmnum, 0, (i + (4096/16)*pwmnum) % 4096 );
|
||||
}
|
||||
#ifdef ESP8266
|
||||
yield(); // take a breather, required for ESP8266
|
||||
#endif
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,83 @@
|
||||
/***************************************************
|
||||
This is an example for our Adafruit 16-channel PWM & Servo driver
|
||||
Servo test - this will drive 8 servos, one after the other on the
|
||||
first 8 pins of the PCA9685
|
||||
|
||||
Pick one up today in the adafruit shop!
|
||||
------> http://www.adafruit.com/products/815
|
||||
|
||||
These drivers use I2C to communicate, 2 pins are required to
|
||||
interface.
|
||||
|
||||
Adafruit invests time and resources providing this open source code,
|
||||
please support Adafruit and open-source hardware by purchasing
|
||||
products from Adafruit!
|
||||
|
||||
Written by Limor Fried/Ladyada for Adafruit Industries.
|
||||
BSD license, all text above must be included in any redistribution
|
||||
****************************************************/
|
||||
|
||||
#include <Wire.h>
|
||||
#include <Adafruit_PWMServoDriver.h>
|
||||
|
||||
// called this way, it uses the default address 0x40
|
||||
Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver();
|
||||
// you can also call it with a different address you want
|
||||
//Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver(0x41);
|
||||
// you can also call it with a different address and I2C interface
|
||||
//Adafruit_PWMServoDriver pwm = Adafruit_PWMServoDriver(&Wire, 0x40);
|
||||
|
||||
// Depending on your servo make, the pulse width min and max may vary, you
|
||||
// want these to be as small/large as possible without hitting the hard stop
|
||||
// for max range. You'll have to tweak them as necessary to match the servos you
|
||||
// have!
|
||||
#define SERVOMIN 150 // this is the 'minimum' pulse length count (out of 4096)
|
||||
#define SERVOMAX 600 // this is the 'maximum' pulse length count (out of 4096)
|
||||
|
||||
// our servo # counter
|
||||
uint8_t servonum = 0;
|
||||
|
||||
void setup() {
|
||||
Serial.begin(9600);
|
||||
Serial.println("8 channel Servo test!");
|
||||
|
||||
pwm.begin();
|
||||
|
||||
pwm.setPWMFreq(60); // Analog servos run at ~60 Hz updates
|
||||
|
||||
delay(10);
|
||||
}
|
||||
|
||||
// you can use this function if you'd like to set the pulse length in seconds
|
||||
// e.g. setServoPulse(0, 0.001) is a ~1 millisecond pulse width. its not precise!
|
||||
void setServoPulse(uint8_t n, double pulse) {
|
||||
double pulselength;
|
||||
|
||||
pulselength = 1000000; // 1,000,000 us per second
|
||||
pulselength /= 60; // 60 Hz
|
||||
Serial.print(pulselength); Serial.println(" us per period");
|
||||
pulselength /= 4096; // 12 bits of resolution
|
||||
Serial.print(pulselength); Serial.println(" us per bit");
|
||||
pulse *= 1000000; // convert to us
|
||||
pulse /= pulselength;
|
||||
Serial.println(pulse);
|
||||
pwm.setPWM(n, 0, pulse);
|
||||
}
|
||||
|
||||
void loop() {
|
||||
// Drive each servo one at a time
|
||||
Serial.println(servonum);
|
||||
for (uint16_t pulselen = SERVOMIN; pulselen < SERVOMAX; pulselen++) {
|
||||
pwm.setPWM(servonum, 0, pulselen);
|
||||
}
|
||||
|
||||
delay(500);
|
||||
for (uint16_t pulselen = SERVOMAX; pulselen > SERVOMIN; pulselen--) {
|
||||
pwm.setPWM(servonum, 0, pulselen);
|
||||
}
|
||||
|
||||
delay(500);
|
||||
|
||||
servonum ++;
|
||||
if (servonum > 7) servonum = 0;
|
||||
}
|
||||
@@ -0,0 +1,9 @@
|
||||
name=Adafruit PWM Servo Driver Library
|
||||
version=1.0.2
|
||||
author=Adafruit
|
||||
maintainer=Adafruit <info@adafruit.com>
|
||||
sentence=Adafruit PWM Servo Driver Library
|
||||
paragraph=Adafruit PWM Servo Driver Library
|
||||
category=Device Control
|
||||
url=https://github.com/adafruit/Adafruit-PWM-Servo-Driver-Library
|
||||
architectures=*
|
||||
26
libraries/Adafruit_PWM_Servo_Driver_Library/license.txt
Normal file
26
libraries/Adafruit_PWM_Servo_Driver_Library/license.txt
Normal file
@@ -0,0 +1,26 @@
|
||||
Software License Agreement (BSD License)
|
||||
|
||||
Copyright (c) 2012, Adafruit Industries
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright
|
||||
notice, this list of conditions and the following disclaimer in the
|
||||
documentation and/or other materials provided with the distribution.
|
||||
3. Neither the name of the copyright holders nor the
|
||||
names of its contributors may be used to endorse or promote products
|
||||
derived from this software without specific prior written permission.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY
|
||||
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
|
||||
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
Reference in New Issue
Block a user