mirror of
https://github.com/brendanhaines/RasPi.git
synced 2025-04-18 23:08:04 -06:00
Adds TCP/IP stuff
This commit is contained in:
parent
6276473f88
commit
7b862ad270
408
IMU/I2Cdev.h
408
IMU/I2Cdev.h
@ -1,4 +1,4 @@
|
|||||||
// I2Cdev library collection - Main I2C device class
|
// I2Cdev library collection - Main I2C device class header file
|
||||||
// Abstracts bit and byte I2C R/W functions into a convenient class
|
// Abstracts bit and byte I2C R/W functions into a convenient class
|
||||||
// 6/9/2012 by Jeff Rowberg <jeff@rowberg.net>
|
// 6/9/2012 by Jeff Rowberg <jeff@rowberg.net>
|
||||||
//
|
//
|
||||||
@ -41,387 +41,37 @@ THE SOFTWARE.
|
|||||||
===============================================
|
===============================================
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdio.h>
|
#ifndef _I2CDEV_H_
|
||||||
#include <stdint.h>
|
#define _I2CDEV_H_
|
||||||
#include <stdlib.h>
|
|
||||||
#include <fcntl.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <errno.h>
|
|
||||||
#include <sys/ioctl.h>
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <linux/i2c-dev.h>
|
|
||||||
#include "I2Cdev.h"
|
|
||||||
|
|
||||||
/** Default constructor.
|
#ifndef TRUE
|
||||||
*/
|
#define TRUE (1==1)
|
||||||
I2Cdev::I2Cdev() {
|
#define FALSE (0==1)
|
||||||
}
|
#endif
|
||||||
|
|
||||||
/** Read a single bit from an 8-bit device register.
|
class I2Cdev {
|
||||||
* @param devAddr I2C slave device address
|
public:
|
||||||
* @param regAddr Register regAddr to read from
|
I2Cdev();
|
||||||
* @param bitNum Bit position to read (0-7)
|
|
||||||
* @param data Container for single bit value
|
|
||||||
* @param timeout Optional read timeout in milliseconds (0 to disable, leave off to use default class value in I2Cdev::readTimeout)
|
|
||||||
* @return Status of read operation (true = success)
|
|
||||||
*/
|
|
||||||
int8_t I2Cdev::readBit(uint8_t devAddr, uint8_t regAddr, uint8_t bitNum, uint8_t *data, uint16_t timeout) {
|
|
||||||
uint8_t b;
|
|
||||||
uint8_t count = readByte(devAddr, regAddr, &b, timeout);
|
|
||||||
*data = b & (1 << bitNum);
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Read a single bit from a 16-bit device register.
|
static int8_t readBit(uint8_t devAddr, uint8_t regAddr, uint8_t bitNum, uint8_t *data, uint16_t timeout=I2Cdev::readTimeout);
|
||||||
* @param devAddr I2C slave device address
|
static int8_t readBitW(uint8_t devAddr, uint8_t regAddr, uint8_t bitNum, uint16_t *data, uint16_t timeout=I2Cdev::readTimeout);
|
||||||
* @param regAddr Register regAddr to read from
|
static int8_t readBits(uint8_t devAddr, uint8_t regAddr, uint8_t bitStart, uint8_t length, uint8_t *data, uint16_t timeout=I2Cdev::readTimeout);
|
||||||
* @param bitNum Bit position to read (0-15)
|
static int8_t readBitsW(uint8_t devAddr, uint8_t regAddr, uint8_t bitStart, uint8_t length, uint16_t *data, uint16_t timeout=I2Cdev::readTimeout);
|
||||||
* @param data Container for single bit value
|
static int8_t readByte(uint8_t devAddr, uint8_t regAddr, uint8_t *data, uint16_t timeout=I2Cdev::readTimeout);
|
||||||
* @param timeout Optional read timeout in milliseconds (0 to disable, leave off to use default class value in I2Cdev::readTimeout)
|
static int8_t readWord(uint8_t devAddr, uint8_t regAddr, uint16_t *data, uint16_t timeout=I2Cdev::readTimeout);
|
||||||
* @return Status of read operation (true = success)
|
static int8_t readBytes(uint8_t devAddr, uint8_t regAddr, uint8_t length, uint8_t *data, uint16_t timeout=I2Cdev::readTimeout);
|
||||||
*/
|
static int8_t readWords(uint8_t devAddr, uint8_t regAddr, uint8_t length, uint16_t *data, uint16_t timeout=I2Cdev::readTimeout);
|
||||||
int8_t I2Cdev::readBitW(uint8_t devAddr, uint8_t regAddr, uint8_t bitNum, uint16_t *data, uint16_t timeout) {
|
|
||||||
uint16_t b;
|
|
||||||
uint8_t count = readWord(devAddr, regAddr, &b, timeout);
|
|
||||||
*data = b & (1 << bitNum);
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Read multiple bits from an 8-bit device register.
|
static bool writeBit(uint8_t devAddr, uint8_t regAddr, uint8_t bitNum, uint8_t data);
|
||||||
* @param devAddr I2C slave device address
|
static bool writeBitW(uint8_t devAddr, uint8_t regAddr, uint8_t bitNum, uint16_t data);
|
||||||
* @param regAddr Register regAddr to read from
|
static bool writeBits(uint8_t devAddr, uint8_t regAddr, uint8_t bitStart, uint8_t length, uint8_t data);
|
||||||
* @param bitStart First bit position to read (0-7)
|
static bool writeBitsW(uint8_t devAddr, uint8_t regAddr, uint8_t bitStart, uint8_t length, uint16_t data);
|
||||||
* @param length Number of bits to read (not more than 8)
|
static bool writeByte(uint8_t devAddr, uint8_t regAddr, uint8_t data);
|
||||||
* @param data Container for right-aligned value (i.e. '101' read from any bitStart position will equal 0x05)
|
static bool writeWord(uint8_t devAddr, uint8_t regAddr, uint16_t data);
|
||||||
* @param timeout Optional read timeout in milliseconds (0 to disable, leave off to use default class value in I2Cdev::readTimeout)
|
static bool writeBytes(uint8_t devAddr, uint8_t regAddr, uint8_t length, uint8_t *data);
|
||||||
* @return Status of read operation (true = success)
|
static bool writeWords(uint8_t devAddr, uint8_t regAddr, uint8_t length, uint16_t *data);
|
||||||
*/
|
|
||||||
int8_t I2Cdev::readBits(uint8_t devAddr, uint8_t regAddr, uint8_t bitStart, uint8_t length, uint8_t *data, uint16_t timeout) {
|
|
||||||
// 01101001 read byte
|
|
||||||
// 76543210 bit numbers
|
|
||||||
// xxx args: bitStart=4, length=3
|
|
||||||
// 010 masked
|
|
||||||
// -> 010 shifted
|
|
||||||
uint8_t count, b;
|
|
||||||
if ((count = readByte(devAddr, regAddr, &b, timeout)) != 0) {
|
|
||||||
uint8_t mask = ((1 << length) - 1) << (bitStart - length + 1);
|
|
||||||
b &= mask;
|
|
||||||
b >>= (bitStart - length + 1);
|
|
||||||
*data = b;
|
|
||||||
}
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Read multiple bits from a 16-bit device register.
|
static uint16_t readTimeout;
|
||||||
* @param devAddr I2C slave device address
|
};
|
||||||
* @param regAddr Register regAddr to read from
|
|
||||||
* @param bitStart First bit position to read (0-15)
|
|
||||||
* @param length Number of bits to read (not more than 16)
|
|
||||||
* @param data Container for right-aligned value (i.e. '101' read from any bitStart position will equal 0x05)
|
|
||||||
* @param timeout Optional read timeout in milliseconds (0 to disable, leave off to use default class value in I2Cdev::readTimeout)
|
|
||||||
* @return Status of read operation (1 = success, 0 = failure, -1 = timeout)
|
|
||||||
*/
|
|
||||||
int8_t I2Cdev::readBitsW(uint8_t devAddr, uint8_t regAddr, uint8_t bitStart, uint8_t length, uint16_t *data, uint16_t timeout) {
|
|
||||||
// 1101011001101001 read byte
|
|
||||||
// fedcba9876543210 bit numbers
|
|
||||||
// xxx args: bitStart=12, length=3
|
|
||||||
// 010 masked
|
|
||||||
// -> 010 shifted
|
|
||||||
uint8_t count;
|
|
||||||
uint16_t w;
|
|
||||||
if ((count = readWord(devAddr, regAddr, &w, timeout)) != 0) {
|
|
||||||
uint16_t mask = ((1 << length) - 1) << (bitStart - length + 1);
|
|
||||||
w &= mask;
|
|
||||||
w >>= (bitStart - length + 1);
|
|
||||||
*data = w;
|
|
||||||
}
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Read single byte from an 8-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr Register regAddr to read from
|
|
||||||
* @param data Container for byte value read from device
|
|
||||||
* @param timeout Optional read timeout in milliseconds (0 to disable, leave off to use default class value in I2Cdev::readTimeout)
|
|
||||||
* @return Status of read operation (true = success)
|
|
||||||
*/
|
|
||||||
int8_t I2Cdev::readByte(uint8_t devAddr, uint8_t regAddr, uint8_t *data, uint16_t timeout) {
|
|
||||||
return readBytes(devAddr, regAddr, 1, data, timeout);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Read single word from a 16-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr Register regAddr to read from
|
|
||||||
* @param data Container for word value read from device
|
|
||||||
* @param timeout Optional read timeout in milliseconds (0 to disable, leave off to use default class value in I2Cdev::readTimeout)
|
|
||||||
* @return Status of read operation (true = success)
|
|
||||||
*/
|
|
||||||
int8_t I2Cdev::readWord(uint8_t devAddr, uint8_t regAddr, uint16_t *data, uint16_t timeout) {
|
|
||||||
return readWords(devAddr, regAddr, 1, data, timeout);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Read multiple bytes from an 8-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr First register regAddr to read from
|
|
||||||
* @param length Number of bytes to read
|
|
||||||
* @param data Buffer to store read data in
|
|
||||||
* @param timeout Optional read timeout in milliseconds (0 to disable, leave off to use default class value in I2Cdev::readTimeout)
|
|
||||||
* @return Number of bytes read (-1 indicates failure)
|
|
||||||
*/
|
|
||||||
int8_t I2Cdev::readBytes(uint8_t devAddr, uint8_t regAddr, uint8_t length, uint8_t *data, uint16_t timeout) {
|
|
||||||
int8_t count = 0;
|
|
||||||
int fd = open("/dev/i2c-1", O_RDWR);
|
|
||||||
|
|
||||||
if (fd < 0) {
|
|
||||||
fprintf(stderr, "Failed to open device: %s\n", strerror(errno));
|
|
||||||
return(-1);
|
|
||||||
}
|
|
||||||
if (ioctl(fd, I2C_SLAVE, devAddr) < 0) {
|
|
||||||
fprintf(stderr, "Failed to select device: %s\n", strerror(errno));
|
|
||||||
close(fd);
|
|
||||||
return(-1);
|
|
||||||
}
|
|
||||||
if (write(fd, ®Addr, 1) != 1) {
|
|
||||||
fprintf(stderr, "Failed to write reg: %s\n", strerror(errno));
|
|
||||||
close(fd);
|
|
||||||
return(-1);
|
|
||||||
}
|
|
||||||
count = read(fd, data, length);
|
|
||||||
if (count < 0) {
|
|
||||||
fprintf(stderr, "Failed to read device(%d): %s\n", count, ::strerror(errno));
|
|
||||||
close(fd);
|
|
||||||
return(-1);
|
|
||||||
} else if (count != length) {
|
|
||||||
fprintf(stderr, "Short read from device, expected %d, got %d\n", length, count);
|
|
||||||
close(fd);
|
|
||||||
return(-1);
|
|
||||||
}
|
|
||||||
close(fd);
|
|
||||||
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Read multiple words from a 16-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr First register regAddr to read from
|
|
||||||
* @param length Number of words to read
|
|
||||||
* @param data Buffer to store read data in
|
|
||||||
* @param timeout Optional read timeout in milliseconds (0 to disable, leave off to use default class value in I2Cdev::readTimeout)
|
|
||||||
* @return Number of words read (0 indicates failure)
|
|
||||||
*/
|
|
||||||
int8_t I2Cdev::readWords(uint8_t devAddr, uint8_t regAddr, uint8_t length, uint16_t *data, uint16_t timeout) {
|
|
||||||
int8_t count = 0;
|
|
||||||
|
|
||||||
printf("ReadWords() not implemented\n");
|
|
||||||
// Use readBytes() and potential byteswap
|
|
||||||
*data = 0; // keep the compiler quiet
|
|
||||||
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** write a single bit in an 8-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr Register regAddr to write to
|
|
||||||
* @param bitNum Bit position to write (0-7)
|
|
||||||
* @param value New bit value to write
|
|
||||||
* @return Status of operation (true = success)
|
|
||||||
*/
|
|
||||||
bool I2Cdev::writeBit(uint8_t devAddr, uint8_t regAddr, uint8_t bitNum, uint8_t data) {
|
|
||||||
uint8_t b;
|
|
||||||
readByte(devAddr, regAddr, &b);
|
|
||||||
b = (data != 0) ? (b | (1 << bitNum)) : (b & ~(1 << bitNum));
|
|
||||||
return writeByte(devAddr, regAddr, b);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** write a single bit in a 16-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr Register regAddr to write to
|
|
||||||
* @param bitNum Bit position to write (0-15)
|
|
||||||
* @param value New bit value to write
|
|
||||||
* @return Status of operation (true = success)
|
|
||||||
*/
|
|
||||||
bool I2Cdev::writeBitW(uint8_t devAddr, uint8_t regAddr, uint8_t bitNum, uint16_t data) {
|
|
||||||
uint16_t w;
|
|
||||||
readWord(devAddr, regAddr, &w);
|
|
||||||
w = (data != 0) ? (w | (1 << bitNum)) : (w & ~(1 << bitNum));
|
|
||||||
return writeWord(devAddr, regAddr, w);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Write multiple bits in an 8-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr Register regAddr to write to
|
|
||||||
* @param bitStart First bit position to write (0-7)
|
|
||||||
* @param length Number of bits to write (not more than 8)
|
|
||||||
* @param data Right-aligned value to write
|
|
||||||
* @return Status of operation (true = success)
|
|
||||||
*/
|
|
||||||
bool I2Cdev::writeBits(uint8_t devAddr, uint8_t regAddr, uint8_t bitStart, uint8_t length, uint8_t data) {
|
|
||||||
// 010 value to write
|
|
||||||
// 76543210 bit numbers
|
|
||||||
// xxx args: bitStart=4, length=3
|
|
||||||
// 00011100 mask byte
|
|
||||||
// 10101111 original value (sample)
|
|
||||||
// 10100011 original & ~mask
|
|
||||||
// 10101011 masked | value
|
|
||||||
uint8_t b;
|
|
||||||
if (readByte(devAddr, regAddr, &b) != 0) {
|
|
||||||
uint8_t mask = ((1 << length) - 1) << (bitStart - length + 1);
|
|
||||||
data <<= (bitStart - length + 1); // shift data into correct position
|
|
||||||
data &= mask; // zero all non-important bits in data
|
|
||||||
b &= ~(mask); // zero all important bits in existing byte
|
|
||||||
b |= data; // combine data with existing byte
|
|
||||||
return writeByte(devAddr, regAddr, b);
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Write multiple bits in a 16-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr Register regAddr to write to
|
|
||||||
* @param bitStart First bit position to write (0-15)
|
|
||||||
* @param length Number of bits to write (not more than 16)
|
|
||||||
* @param data Right-aligned value to write
|
|
||||||
* @return Status of operation (true = success)
|
|
||||||
*/
|
|
||||||
bool I2Cdev::writeBitsW(uint8_t devAddr, uint8_t regAddr, uint8_t bitStart, uint8_t length, uint16_t data) {
|
|
||||||
// 010 value to write
|
|
||||||
// fedcba9876543210 bit numbers
|
|
||||||
// xxx args: bitStart=12, length=3
|
|
||||||
// 0001110000000000 mask byte
|
|
||||||
// 1010111110010110 original value (sample)
|
|
||||||
// 1010001110010110 original & ~mask
|
|
||||||
// 1010101110010110 masked | value
|
|
||||||
uint16_t w;
|
|
||||||
if (readWord(devAddr, regAddr, &w) != 0) {
|
|
||||||
uint8_t mask = ((1 << length) - 1) << (bitStart - length + 1);
|
|
||||||
data <<= (bitStart - length + 1); // shift data into correct position
|
|
||||||
data &= mask; // zero all non-important bits in data
|
|
||||||
w &= ~(mask); // zero all important bits in existing word
|
|
||||||
w |= data; // combine data with existing word
|
|
||||||
return writeWord(devAddr, regAddr, w);
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Write single byte to an 8-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr Register address to write to
|
|
||||||
* @param data New byte value to write
|
|
||||||
* @return Status of operation (true = success)
|
|
||||||
*/
|
|
||||||
bool I2Cdev::writeByte(uint8_t devAddr, uint8_t regAddr, uint8_t data) {
|
|
||||||
return writeBytes(devAddr, regAddr, 1, &data);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Write single word to a 16-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr Register address to write to
|
|
||||||
* @param data New word value to write
|
|
||||||
* @return Status of operation (true = success)
|
|
||||||
*/
|
|
||||||
bool I2Cdev::writeWord(uint8_t devAddr, uint8_t regAddr, uint16_t data) {
|
|
||||||
return writeWords(devAddr, regAddr, 1, &data);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Write multiple bytes to an 8-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr First register address to write to
|
|
||||||
* @param length Number of bytes to write
|
|
||||||
* @param data Buffer to copy new data from
|
|
||||||
* @return Status of operation (true = success)
|
|
||||||
*/
|
|
||||||
bool I2Cdev::writeBytes(uint8_t devAddr, uint8_t regAddr, uint8_t length, uint8_t* data) {
|
|
||||||
int8_t count = 0;
|
|
||||||
uint8_t buf[128];
|
|
||||||
int fd;
|
|
||||||
|
|
||||||
if (length > 127) {
|
|
||||||
fprintf(stderr, "Byte write count (%d) > 127\n", length);
|
|
||||||
return(FALSE);
|
|
||||||
}
|
|
||||||
|
|
||||||
fd = open("/dev/i2c-1", O_RDWR);
|
|
||||||
if (fd < 0) {
|
|
||||||
fprintf(stderr, "Failed to open device: %s\n", strerror(errno));
|
|
||||||
return(FALSE);
|
|
||||||
}
|
|
||||||
if (ioctl(fd, I2C_SLAVE, devAddr) < 0) {
|
|
||||||
fprintf(stderr, "Failed to select device: %s\n", strerror(errno));
|
|
||||||
close(fd);
|
|
||||||
return(FALSE);
|
|
||||||
}
|
|
||||||
buf[0] = regAddr;
|
|
||||||
memcpy(buf+1,data,length);
|
|
||||||
count = write(fd, buf, length+1);
|
|
||||||
if (count < 0) {
|
|
||||||
fprintf(stderr, "Failed to write device(%d): %s\n", count, ::strerror(errno));
|
|
||||||
close(fd);
|
|
||||||
return(FALSE);
|
|
||||||
} else if (count != length+1) {
|
|
||||||
fprintf(stderr, "Short write to device, expected %d, got %d\n", length+1, count);
|
|
||||||
close(fd);
|
|
||||||
return(FALSE);
|
|
||||||
}
|
|
||||||
close(fd);
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Write multiple words to a 16-bit device register.
|
|
||||||
* @param devAddr I2C slave device address
|
|
||||||
* @param regAddr First register address to write to
|
|
||||||
* @param length Number of words to write
|
|
||||||
* @param data Buffer to copy new data from
|
|
||||||
* @return Status of operation (true = success)
|
|
||||||
*/
|
|
||||||
bool I2Cdev::writeWords(uint8_t devAddr, uint8_t regAddr, uint8_t length, uint16_t* data) {
|
|
||||||
int8_t count = 0;
|
|
||||||
uint8_t buf[128];
|
|
||||||
int i, fd;
|
|
||||||
|
|
||||||
// Should do potential byteswap and call writeBytes() really, but that
|
|
||||||
// messes with the callers buffer
|
|
||||||
|
|
||||||
if (length > 63) {
|
|
||||||
fprintf(stderr, "Word write count (%d) > 63\n", length);
|
|
||||||
return(FALSE);
|
|
||||||
}
|
|
||||||
|
|
||||||
fd = open("/dev/i2c-1", O_RDWR);
|
|
||||||
if (fd < 0) {
|
|
||||||
fprintf(stderr, "Failed to open device: %s\n", strerror(errno));
|
|
||||||
return(FALSE);
|
|
||||||
}
|
|
||||||
if (ioctl(fd, I2C_SLAVE, devAddr) < 0) {
|
|
||||||
fprintf(stderr, "Failed to select device: %s\n", strerror(errno));
|
|
||||||
close(fd);
|
|
||||||
return(FALSE);
|
|
||||||
}
|
|
||||||
buf[0] = regAddr;
|
|
||||||
for (i = 0; i < length; i++) {
|
|
||||||
buf[i*2+1] = data[i] >> 8;
|
|
||||||
buf[i*2+2] = data[i];
|
|
||||||
}
|
|
||||||
count = write(fd, buf, length*2+1);
|
|
||||||
if (count < 0) {
|
|
||||||
fprintf(stderr, "Failed to write device(%d): %s\n", count, ::strerror(errno));
|
|
||||||
close(fd);
|
|
||||||
return(FALSE);
|
|
||||||
} else if (count != length*2+1) {
|
|
||||||
fprintf(stderr, "Short write to device, expected %d, got %d\n", length+1, count);
|
|
||||||
close(fd);
|
|
||||||
return(FALSE);
|
|
||||||
}
|
|
||||||
close(fd);
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Default timeout value for read operations.
|
|
||||||
* Set this to 0 to disable timeout detection.
|
|
||||||
*/
|
|
||||||
uint16_t I2Cdev::readTimeout = 0;
|
|
||||||
|
|
||||||
|
#endif /* _I2CDEV_H_ */
|
||||||
|
12
TCP/Makefile
Normal file
12
TCP/Makefile
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
all: TCPserver.o TCPserverExample
|
||||||
|
|
||||||
|
HDRS = TCPserver.h
|
||||||
|
CXXFLAGS = -Wall
|
||||||
|
|
||||||
|
TCPserver.o TCPserverExample.o: $(HDRS)
|
||||||
|
|
||||||
|
TCPserverExample: TCPserver.o TCPserverExample.o
|
||||||
|
$(CXX) $^ -o $@
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm TCPserver.o TCPserverExample.o TCPserverExample
|
BIN
TCP/TCPclient.class
Normal file
BIN
TCP/TCPclient.class
Normal file
Binary file not shown.
68
TCP/TCPclient.java
Normal file
68
TCP/TCPclient.java
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
/**
|
||||||
|
* TCPclient.java
|
||||||
|
* Written by Brendan Haines
|
||||||
|
* Based on example from http://www.java-samples.com/showtutorial.php?tutorialid=1167
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.lang.*;
|
||||||
|
import java.io.*;
|
||||||
|
import java.net.*;
|
||||||
|
|
||||||
|
class TCPclient {
|
||||||
|
/** Socket at the server */
|
||||||
|
private Socket socket;
|
||||||
|
/** buffer to hold received info */
|
||||||
|
private BufferedReader buffer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Connects to socket at server
|
||||||
|
* @param host - the server's IP
|
||||||
|
* @param port - the port number of the socket at the server
|
||||||
|
*/
|
||||||
|
public TCPclient( String host, int port ) {
|
||||||
|
try {
|
||||||
|
System.out.println( "setting up socket" );
|
||||||
|
socket = new Socket( host, port );
|
||||||
|
System.out.println( "socket set up\nsetting up buffer" );
|
||||||
|
buffer = new BufferedReader( new InputStreamReader( socket.getInputStream() ) );
|
||||||
|
System.out.println( "buffer set up" );
|
||||||
|
}
|
||||||
|
catch( Exception e ) {
|
||||||
|
System.out.println( "ERROR -- constructor" );
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* reads one line from the receive buffer
|
||||||
|
* @return one line of the receive buffer on success, null on failure
|
||||||
|
*/
|
||||||
|
public String readLine() {
|
||||||
|
try{
|
||||||
|
if( buffer.ready() ) {
|
||||||
|
return buffer.readLine();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch( Exception e ) {
|
||||||
|
System.out.println( "Hmmm... it won't read" );
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* close connection
|
||||||
|
*/
|
||||||
|
public void close() {
|
||||||
|
try {
|
||||||
|
buffer.close();
|
||||||
|
socket.close();
|
||||||
|
}
|
||||||
|
catch( Exception e ) {
|
||||||
|
System.out.println( "SH*T! the connection won't close" );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
BIN
TCP/TCPclientExample.class
Normal file
BIN
TCP/TCPclientExample.class
Normal file
Binary file not shown.
11
TCP/TCPclientExample.java
Normal file
11
TCP/TCPclientExample.java
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
/**
|
||||||
|
* Written By Brendan Haines
|
||||||
|
*/
|
||||||
|
|
||||||
|
class TCPclientExample {
|
||||||
|
public static void main( String args[] ) {
|
||||||
|
TCPclient client = new TCPclient( "192.168.42.1", 51719 );
|
||||||
|
System.out.println( "Setup successful" );
|
||||||
|
client.close();
|
||||||
|
}
|
||||||
|
}
|
134
TCP/TCPserver.cpp
Normal file
134
TCP/TCPserver.cpp
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
/*
|
||||||
|
* TCPserver.cpp
|
||||||
|
* Written by Brendan Haines
|
||||||
|
* Based heavily on example code found at http://www.linuxhowtos.org/C_C++/socket.htm
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
|
|
||||||
|
#include <pthread.h>
|
||||||
|
|
||||||
|
#include "TCPserver.h"
|
||||||
|
|
||||||
|
TCPserver::TCPserver()
|
||||||
|
{}
|
||||||
|
|
||||||
|
// Constructor sets everything up and starts monitoring thread
|
||||||
|
TCPserver::TCPserver( int port )
|
||||||
|
{
|
||||||
|
portNumber = port;
|
||||||
|
setupSocket();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Destructor closes sockets
|
||||||
|
TCPserver::~TCPserver()
|
||||||
|
{
|
||||||
|
close( connectionFD );
|
||||||
|
close( socketFD );
|
||||||
|
}
|
||||||
|
|
||||||
|
// a unified/consistent way of reporting errors
|
||||||
|
void TCPserver::handleError( const char* message )
|
||||||
|
{
|
||||||
|
perror( message ); // print message to stderror
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns true on success, false on failure
|
||||||
|
bool TCPserver::setupSocket()
|
||||||
|
{
|
||||||
|
// Create a new socket
|
||||||
|
// AF_INET: internet domain (not unix domain)
|
||||||
|
// SOCK_STREAM: stream (not datagram)
|
||||||
|
// 0: lets OS decide correct protocol (TCP)
|
||||||
|
socketFD = socket( AF_INET, SOCK_STREAM, 0 );
|
||||||
|
if( socketFD < 0 ) // error setting up socket
|
||||||
|
{
|
||||||
|
handleError( "Error setting up socket\n" );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// fill server_addr with zeroes
|
||||||
|
// NOTE: memset sets arg2 bytes following arg0 to arg1
|
||||||
|
// NOTE: sizeof returns the size of the passed type in bytes
|
||||||
|
memset( (char*) &server_addr, 0, sizeof( server_addr ) );
|
||||||
|
|
||||||
|
server_addr.sin_family = AF_INET; // just following the example (and its documentation)
|
||||||
|
server_addr.sin_port = htons( portNumber ); // assign the port number. NOTE: htons converts host byte order to network byte order
|
||||||
|
server_addr.sin_addr.s_addr = INADDR_ANY; // INADDR_ANY represents the IP address of the host
|
||||||
|
|
||||||
|
// bind: binds the socket to an address
|
||||||
|
if( bind( socketFD, (struct sockaddr*) &server_addr, sizeof( server_addr ) ) < 0 )
|
||||||
|
{
|
||||||
|
handleError( "Error binding socket" );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// listen for connections to socket (max 5 waiting)
|
||||||
|
listen( socketFD, 5 );
|
||||||
|
|
||||||
|
clientAddrLen = sizeof( client_addr );
|
||||||
|
|
||||||
|
// wait for client to connect
|
||||||
|
printf( "Waiting for client to connect\n" );
|
||||||
|
connectionFD = accept( socketFD, (struct sockaddr*) &client_addr, &clientAddrLen );
|
||||||
|
if( connectionFD < 0 )
|
||||||
|
{
|
||||||
|
handleError( "Error on accept" );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// empty buffer
|
||||||
|
memset( buffer, 0, sizeof( buffer ) );
|
||||||
|
|
||||||
|
return true; // setup successful
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns a pointer to the buffer
|
||||||
|
char* TCPserver::getBuffer()
|
||||||
|
{
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns the connection file descriptor
|
||||||
|
int TCPserver::getConnectionFD()
|
||||||
|
{
|
||||||
|
return connectionFD;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
void TCPserver::startMonitoringThread()
|
||||||
|
{
|
||||||
|
// create thread: ( thread, NULL = default values, function to run, parameter (void*) or NULL )
|
||||||
|
if( pthread_create( &monitorThread, NULL, monitoringThread, (void*) this ) )
|
||||||
|
{
|
||||||
|
// error creating thread
|
||||||
|
handleError( "Error creating thread" );
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void* TCPserver::monitoringThread( void* serverPointer )
|
||||||
|
{
|
||||||
|
int n; // contains the number of characters read
|
||||||
|
|
||||||
|
TCPserver server;
|
||||||
|
|
||||||
|
while( true )
|
||||||
|
{
|
||||||
|
// read from the socket and put in the buffer
|
||||||
|
n = read( server.getConnectionFD(), server.getBuffer(), 255 );
|
||||||
|
if( n < 0 )
|
||||||
|
{
|
||||||
|
server.handleError( "Error reading -- monitorThread " );
|
||||||
|
pthread_exit( NULL ); // exit this thread
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
42
TCP/TCPserver.h
Normal file
42
TCP/TCPserver.h
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
/*
|
||||||
|
* TCPserver.h
|
||||||
|
* Written by Brendan Haines
|
||||||
|
* Based heavily on example code found at http://www.linuxhowtos.org/C_C++/socket.htm
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef TCP_SERVER_H
|
||||||
|
#define TCP_SERVER_H
|
||||||
|
|
||||||
|
#include <netinet/in.h>
|
||||||
|
|
||||||
|
|
||||||
|
class TCPserver {
|
||||||
|
public:
|
||||||
|
TCPserver();
|
||||||
|
TCPserver( int port ); // Constructor
|
||||||
|
~TCPserver(); // Destructor
|
||||||
|
|
||||||
|
void handleError( const char* message );
|
||||||
|
|
||||||
|
char* getBuffer();
|
||||||
|
int getConnectionFD();
|
||||||
|
|
||||||
|
char buffer[ 256 ]; // buffer to store read data
|
||||||
|
private:
|
||||||
|
int socketFD; // file descriptor for the socket
|
||||||
|
int connectionFD; // file descriptor for the connection
|
||||||
|
int portNumber; // port number of the socket
|
||||||
|
socklen_t clientAddrLen; // stores the length of the client's address
|
||||||
|
|
||||||
|
struct sockaddr_in server_addr; // contains the server's address
|
||||||
|
struct sockaddr_in client_addr; // contains the client's address
|
||||||
|
|
||||||
|
//pthread_t monitorThread; // thread to monitor the socket and read to buffer
|
||||||
|
|
||||||
|
bool setupSocket();
|
||||||
|
|
||||||
|
//void startMonitoringThread();
|
||||||
|
//void* monitoringThread( void* );
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
BIN
TCP/TCPserver.o
Normal file
BIN
TCP/TCPserver.o
Normal file
Binary file not shown.
BIN
TCP/TCPserverExample
Executable file
BIN
TCP/TCPserverExample
Executable file
Binary file not shown.
11
TCP/TCPserverExample.cpp
Normal file
11
TCP/TCPserverExample.cpp
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
/*
|
||||||
|
* TCPserverExample.cpp
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "stdio.h"
|
||||||
|
#include "TCPserver.h"
|
||||||
|
|
||||||
|
int main( int argc, char* argv[] )
|
||||||
|
{
|
||||||
|
TCPserver serv( 51719 );
|
||||||
|
}
|
BIN
TCP/TCPserverExample.o
Normal file
BIN
TCP/TCPserverExample.o
Normal file
Binary file not shown.
5
TCPexample/README
Normal file
5
TCPexample/README
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
Example code using TCP/IP protocol with C++
|
||||||
|
Example and documentation found at http://www.linuxhowtos.org/C_C++/socket.htm
|
||||||
|
|
||||||
|
server.c and client.c are original files
|
||||||
|
server2.c is commented and reformatted to facilitate understanding the underlying process
|
BIN
TCPexample/client
Executable file
BIN
TCPexample/client
Executable file
Binary file not shown.
57
TCPexample/client.c
Normal file
57
TCPexample/client.c
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
|
#include <netdb.h>
|
||||||
|
|
||||||
|
void error(const char *msg)
|
||||||
|
{
|
||||||
|
perror(msg);
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
int sockfd, portno, n;
|
||||||
|
struct sockaddr_in serv_addr;
|
||||||
|
struct hostent *server;
|
||||||
|
|
||||||
|
char buffer[256];
|
||||||
|
if (argc < 3) {
|
||||||
|
fprintf(stderr,"usage %s hostname port\n", argv[0]);
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
portno = atoi(argv[2]);
|
||||||
|
sockfd = socket(AF_INET, SOCK_STREAM, 0);
|
||||||
|
if (sockfd < 0)
|
||||||
|
error("ERROR opening socket");
|
||||||
|
server = gethostbyname(argv[1]);
|
||||||
|
if (server == NULL) {
|
||||||
|
fprintf(stderr,"ERROR, no such host\n");
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
bzero((char *) &serv_addr, sizeof(serv_addr));
|
||||||
|
serv_addr.sin_family = AF_INET;
|
||||||
|
bcopy((char *)server->h_addr,
|
||||||
|
(char *)&serv_addr.sin_addr.s_addr,
|
||||||
|
server->h_length);
|
||||||
|
serv_addr.sin_port = htons(portno);
|
||||||
|
if (connect(sockfd,(struct sockaddr *) &serv_addr,sizeof(serv_addr)) < 0)
|
||||||
|
error("ERROR connecting");
|
||||||
|
printf("Please enter the message: ");
|
||||||
|
bzero(buffer,256);
|
||||||
|
fgets(buffer,255,stdin);
|
||||||
|
n = write(sockfd,buffer,strlen(buffer));
|
||||||
|
if (n < 0)
|
||||||
|
error("ERROR writing to socket");
|
||||||
|
bzero(buffer,256);
|
||||||
|
n = read(sockfd,buffer,255);
|
||||||
|
if (n < 0)
|
||||||
|
error("ERROR reading from socket");
|
||||||
|
printf("%s\n",buffer);
|
||||||
|
close(sockfd);
|
||||||
|
return 0;
|
||||||
|
}
|
BIN
TCPexample/server
Executable file
BIN
TCPexample/server
Executable file
Binary file not shown.
55
TCPexample/server.c
Normal file
55
TCPexample/server.c
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
/* A simple server in the internet domain using TCP
|
||||||
|
The port number is passed as an argument */
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
|
|
||||||
|
void error(const char *msg)
|
||||||
|
{
|
||||||
|
perror(msg);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
int sockfd, newsockfd, portno;
|
||||||
|
socklen_t clilen;
|
||||||
|
char buffer[256];
|
||||||
|
struct sockaddr_in serv_addr, cli_addr;
|
||||||
|
int n;
|
||||||
|
if (argc < 2) {
|
||||||
|
fprintf(stderr,"ERROR, no port provided\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
sockfd = socket(AF_INET, SOCK_STREAM, 0);
|
||||||
|
if (sockfd < 0)
|
||||||
|
error("ERROR opening socket");
|
||||||
|
bzero((char *) &serv_addr, sizeof(serv_addr));
|
||||||
|
portno = atoi(argv[1]);
|
||||||
|
serv_addr.sin_family = AF_INET;
|
||||||
|
serv_addr.sin_addr.s_addr = INADDR_ANY;
|
||||||
|
serv_addr.sin_port = htons(portno);
|
||||||
|
if (bind(sockfd, (struct sockaddr *) &serv_addr,
|
||||||
|
sizeof(serv_addr)) < 0)
|
||||||
|
error("ERROR on binding");
|
||||||
|
listen(sockfd,5);
|
||||||
|
clilen = sizeof(cli_addr);
|
||||||
|
newsockfd = accept(sockfd,
|
||||||
|
(struct sockaddr *) &cli_addr,
|
||||||
|
&clilen);
|
||||||
|
if (newsockfd < 0)
|
||||||
|
error("ERROR on accept");
|
||||||
|
bzero(buffer,256);
|
||||||
|
n = read(newsockfd,buffer,255);
|
||||||
|
if (n < 0) error("ERROR reading from socket");
|
||||||
|
printf("Here is the message: %s\n",buffer);
|
||||||
|
n = write(newsockfd,"I got your message",18);
|
||||||
|
if (n < 0) error("ERROR writing to socket");
|
||||||
|
close(newsockfd);
|
||||||
|
close(sockfd);
|
||||||
|
return 0;
|
||||||
|
}
|
127
TCPexample/server2.c
Normal file
127
TCPexample/server2.c
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
/* A simple server in the internet domain using TCP
|
||||||
|
The port number is passed as an argument */
|
||||||
|
|
||||||
|
// From:
|
||||||
|
// http://www.linuxhowtos.org/C_C++/socket.htm
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
|
|
||||||
|
void error( const char *msg )
|
||||||
|
{
|
||||||
|
perror( msg );
|
||||||
|
exit( 1 );
|
||||||
|
}
|
||||||
|
|
||||||
|
int main( int argc, char *argv[] )
|
||||||
|
{
|
||||||
|
// VARIABLE DEFINITONS
|
||||||
|
|
||||||
|
// file descriptors
|
||||||
|
int sockfd; // returned by socket system call
|
||||||
|
int newsockfd; // returned by accept system call
|
||||||
|
|
||||||
|
int portno; // stores port number (passed as argument)
|
||||||
|
socklen_t clilen; // stores the size of the client address. needed for the accept system call
|
||||||
|
char buffer[ 256 ]; // server reads characters from socket connection into buffer
|
||||||
|
|
||||||
|
// sockaddr_in is defined in "netinet/in.h"
|
||||||
|
struct sockaddr_in serv_addr; // serv_addr contains the server address
|
||||||
|
struct sockaddr_in cli_addr; // cli_addr contains the client address
|
||||||
|
|
||||||
|
int n; // return value for read/write calls (contains the # of characters read/written)
|
||||||
|
|
||||||
|
// BEGIN REAL CODE
|
||||||
|
|
||||||
|
// Check that there are enough arguments
|
||||||
|
if( argc < 2 )
|
||||||
|
{
|
||||||
|
fprintf( stderr, "ERROR, no port provided\n" );
|
||||||
|
exit( 1 );
|
||||||
|
}
|
||||||
|
|
||||||
|
// socket system call creates new socket
|
||||||
|
// arg0: address domain ( Unix domain [AF_UNIX] or Internet domain [AF_INET] )
|
||||||
|
// arg1: socket type ( Stream [SOCK_STREAM] or Datagram [SOCK_DGRAM] )
|
||||||
|
// arg2: protocol ( 0 lets system decide best protocol: TCP for streams, UDP for datagrams )
|
||||||
|
// returns file descriptor on success
|
||||||
|
// returns -1 on failure
|
||||||
|
sockfd = socket( AF_INET, SOCK_STREAM, 0 );
|
||||||
|
if( sockfd < 0 ) // on error: sockfd = -1
|
||||||
|
error( "ERROR opening socket" );
|
||||||
|
|
||||||
|
// bzero sets all values in a buffer to zero
|
||||||
|
// arg0: pointer to buffer
|
||||||
|
// arg1: length of buffer
|
||||||
|
bzero( (char*) &serv_addr, sizeof( serv_addr ) ); // initializes serv_addr to all zeroes
|
||||||
|
|
||||||
|
// atoi converts a string of digits to an integer
|
||||||
|
portno = atoi( argv[ 1 ] );
|
||||||
|
|
||||||
|
serv_addr.sin_family = AF_INET; // sin_family should always be set to symbolic constant AF_INET
|
||||||
|
serv_addr.sin_addr.s_addr = INADDR_ANY; // sin_addr.s_addr contains the IP address of the host. INADDR_ANY is a symbolic constant which get this address
|
||||||
|
serv_addr.sin_port = htons( portno ); // sin_port contains the port number. htons converts a port number in host byte order to network byte order
|
||||||
|
|
||||||
|
// bind system call binds a socket to an address ( address = address of host & port number)
|
||||||
|
// arg0: socket file descriptor
|
||||||
|
// arg1: pointer to structure of type sockaddr. what is passed is a structure of type sockaddr_in, must be cast to correct type
|
||||||
|
// arg2: size of structure &arg1
|
||||||
|
// returns 0 on success ?????????
|
||||||
|
if( bind( sockfd, ( struct sockaddr* ) &serv_addr, sizeof( serv_addr ) ) < 0 )
|
||||||
|
error( "ERROR on binding" );
|
||||||
|
|
||||||
|
// listen system call allows program to listen to socket for connections
|
||||||
|
// arg0: file descriptor of socket
|
||||||
|
// arg1: size of backlog queue (maximum number of connections which can be waiting while handling any specific connection) (standard: 5)
|
||||||
|
listen( sockfd, 5 );
|
||||||
|
|
||||||
|
// set client address length to the size of client's address
|
||||||
|
clilen = sizeof( cli_addr );
|
||||||
|
|
||||||
|
// accept system call causes the program to stop until a client connects to the server
|
||||||
|
// arg0: file descriptor of the socket
|
||||||
|
// arg1: pointer for the client address structure
|
||||||
|
// arg2: size of structure &arg1
|
||||||
|
// returns a file descriptor for the connection on success
|
||||||
|
// returns -1 on failure ?????
|
||||||
|
newsockfd = accept( sockfd, (struct sockaddr *) &cli_addr, &clilen );
|
||||||
|
if( newsockfd < 0 )
|
||||||
|
error( "ERROR on accept" );
|
||||||
|
|
||||||
|
bzero( buffer, 256 ); // clear the receive buffer
|
||||||
|
|
||||||
|
// read system call reads from socket
|
||||||
|
// read will block the program until there is something to read
|
||||||
|
// arg0: file descriptor for the connection (NOTE: this is the new one returned by accept)
|
||||||
|
// arg1: buffer to put characters into
|
||||||
|
// arg2: maximum number of character to read and put in buffer
|
||||||
|
// return number of characters read on success
|
||||||
|
// return -1 on failure ?????
|
||||||
|
n = read( newsockfd, buffer, 255 );
|
||||||
|
if ( n < 0 )
|
||||||
|
error("ERROR reading from socket");
|
||||||
|
|
||||||
|
// buffer will contain the message now
|
||||||
|
printf( "Here is the message: %s\n", buffer );
|
||||||
|
|
||||||
|
// write system call writes to socket
|
||||||
|
// arg0: file descriptor for the connection (NOTE: again, this is the new one returned by accept)
|
||||||
|
// arg1: pointer to character (char*)
|
||||||
|
// arg2: length of &arg1
|
||||||
|
// return number of characters written on success
|
||||||
|
// return -1 on failure
|
||||||
|
n = write( newsockfd, "I got your message", 18 );
|
||||||
|
if ( n < 0 )
|
||||||
|
error( "ERROR writing to socket" );
|
||||||
|
|
||||||
|
// when socket is no longer needed, close the connection
|
||||||
|
close( newsockfd );
|
||||||
|
close( sockfd );
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user