Initial send_data() support.
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@39587 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
297a8ac7d2
commit
5655de51dd
@ -23,12 +23,17 @@
|
|||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <termios.h>
|
#include <termios.h>
|
||||||
|
#include <sys/uio.h>
|
||||||
|
|
||||||
|
|
||||||
#define HDLC_CONTROL_ESCAPE 0x7d
|
|
||||||
#define HDLC_FLAG_SEQUENCE 0x7e
|
#define HDLC_FLAG_SEQUENCE 0x7e
|
||||||
|
#define HDLC_CONTROL_ESCAPE 0x7d
|
||||||
|
|
||||||
#define HDLC_ALL_STATIONS 0xff
|
#define HDLC_ALL_STATIONS 0xff
|
||||||
#define HDLC_UI 0x03
|
#define HDLC_UI 0x03
|
||||||
|
|
||||||
|
#define HDLC_HEADER_LENGTH 4
|
||||||
|
|
||||||
|
|
||||||
enum dialup_state {
|
enum dialup_state {
|
||||||
DOWN,
|
DOWN,
|
||||||
@ -186,7 +191,7 @@ dialup_init(const char* name, net_device** _device)
|
|||||||
if (status < B_OK)
|
if (status < B_OK)
|
||||||
return status;
|
return status;
|
||||||
|
|
||||||
dialup_device* device = new (std::nothrow) dialup_device;
|
dialup_device* device = new (std::nothrow)dialup_device;
|
||||||
if (device == NULL) {
|
if (device == NULL) {
|
||||||
put_module(NET_BUFFER_MODULE_NAME);
|
put_module(NET_BUFFER_MODULE_NAME);
|
||||||
return B_NO_MEMORY;
|
return B_NO_MEMORY;
|
||||||
@ -197,9 +202,9 @@ dialup_init(const char* name, net_device** _device)
|
|||||||
strcpy(device->name, name);
|
strcpy(device->name, name);
|
||||||
device->flags = IFF_POINTOPOINT;
|
device->flags = IFF_POINTOPOINT;
|
||||||
device->type = IFT_PPP; // this device handle RFC 1331 frame format only
|
device->type = IFT_PPP; // this device handle RFC 1331 frame format only
|
||||||
device->mtu = 1502;
|
device->mtu = 1500;
|
||||||
device->media = 0;
|
device->media = 0;
|
||||||
device->header_length = 8; // HDLC_HEADER_LENGTH;
|
device->header_length = HDLC_HEADER_LENGTH;
|
||||||
device->fd = -1;
|
device->fd = -1;
|
||||||
device->state = DOWN;
|
device->state = DOWN;
|
||||||
device->data_mode = false;
|
device->data_mode = false;
|
||||||
@ -224,6 +229,7 @@ dialup_uninit(net_device* _device)
|
|||||||
delete device;
|
delete device;
|
||||||
|
|
||||||
put_module(NET_BUFFER_MODULE_NAME);
|
put_module(NET_BUFFER_MODULE_NAME);
|
||||||
|
gBufferModule = NULL;
|
||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -246,6 +252,8 @@ dialup_up(net_device* _device)
|
|||||||
|
|
||||||
// adjust options
|
// adjust options
|
||||||
device->line_config.c_cflag &= ~CBAUD;
|
device->line_config.c_cflag &= ~CBAUD;
|
||||||
|
device->line_config.c_cflag &= CSIZE;
|
||||||
|
device->line_config.c_cflag &= CS8;
|
||||||
device->line_config.c_cflag |= B115200; // TODO: make this configurable too...
|
device->line_config.c_cflag |= B115200; // TODO: make this configurable too...
|
||||||
device->line_config.c_cflag |= (CLOCAL | CREAD);
|
device->line_config.c_cflag |= (CLOCAL | CREAD);
|
||||||
device->line_config.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG);
|
device->line_config.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG);
|
||||||
@ -258,27 +266,31 @@ dialup_up(net_device* _device)
|
|||||||
sizeof(device->line_config)) < 0)
|
sizeof(device->line_config)) < 0)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
// TODO: init modem & start dialing phase
|
// init modem & start dialing phase
|
||||||
device->state = DIALING;
|
|
||||||
|
|
||||||
char reply[32];
|
char reply[32];
|
||||||
|
|
||||||
// Send modem init string
|
if (strlen(device->init_string) > 0) {
|
||||||
if (send_command(device, device->init_string) != B_OK
|
// Send modem init string
|
||||||
|| read_command_reply(device, device->init_string,
|
if (send_command(device, device->init_string) != B_OK
|
||||||
reply, sizeof(reply)) != B_OK
|
|| read_command_reply(device, device->init_string,
|
||||||
|| strcmp(reply, "OK")) {
|
reply, sizeof(reply)) != B_OK
|
||||||
errno = B_IO_ERROR;
|
|| strcmp(reply, "OK")) {
|
||||||
goto err;
|
errno = B_IO_ERROR;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Send dialing string
|
if (strlen(device->dial_string) > 0) {
|
||||||
if (send_command(device, device->dial_string) != B_OK
|
// Send dialing string
|
||||||
|| read_command_reply(device, device->dial_string,
|
device->state = DIALING;
|
||||||
reply, sizeof(reply)) != B_OK
|
if (send_command(device, device->dial_string) != B_OK
|
||||||
|| strncmp(reply, "CONNECT", 7)) {
|
|| read_command_reply(device, device->dial_string,
|
||||||
errno = B_IO_ERROR;
|
reply, sizeof(reply)) != B_OK
|
||||||
goto err;
|
|| strncmp(reply, "CONNECT", 7)) {
|
||||||
|
errno = B_IO_ERROR;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
device->state = UP;
|
device->state = UP;
|
||||||
@ -334,7 +346,88 @@ dialup_control(net_device* _device, int32 op, void* argument,
|
|||||||
status_t
|
status_t
|
||||||
dialup_send_data(net_device* _device, net_buffer* buffer)
|
dialup_send_data(net_device* _device, net_buffer* buffer)
|
||||||
{
|
{
|
||||||
return B_NOT_SUPPORTED;
|
dialup_device* device = (dialup_device*)_device;
|
||||||
|
|
||||||
|
if (device->fd == -1)
|
||||||
|
return B_FILE_ERROR;
|
||||||
|
|
||||||
|
dprintf("try to send HDLC packet of %lu bytes (flags %ld):\n", buffer->size, buffer->flags);
|
||||||
|
|
||||||
|
if (buffer->size < HDLC_HEADER_LENGTH)
|
||||||
|
return B_BAD_VALUE;
|
||||||
|
|
||||||
|
iovec* ioVectors = NULL;
|
||||||
|
iovec* ioVector;
|
||||||
|
uint8* packet = NULL;
|
||||||
|
int packetSize = 0;
|
||||||
|
status_t status;
|
||||||
|
ssize_t bytesWritten;
|
||||||
|
|
||||||
|
uint32 vectorCount = gBufferModule->count_iovecs(buffer);
|
||||||
|
if (vectorCount < 1) {
|
||||||
|
status = B_BAD_VALUE;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
ioVectors = (iovec*)malloc(sizeof(iovec)*vectorCount);
|
||||||
|
if (ioVectors == NULL) {
|
||||||
|
status = B_NO_MEMORY;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
gBufferModule->get_iovecs(buffer, ioVectors, vectorCount);
|
||||||
|
|
||||||
|
// encode HDLC packet
|
||||||
|
|
||||||
|
// worst case: begin and end sequence flags and each byte needing escape
|
||||||
|
packet = (uint8*) malloc(2 + 2 * buffer->size);
|
||||||
|
if (packet == NULL) {
|
||||||
|
status = B_NO_MEMORY;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
// mark frame start
|
||||||
|
packet[packetSize++] = HDLC_FLAG_SEQUENCE;
|
||||||
|
|
||||||
|
ioVector = ioVectors;
|
||||||
|
while (vectorCount--) {
|
||||||
|
uint8* data = (uint8*) ioVector->iov_base;
|
||||||
|
for (unsigned int i = 0; i < ioVector->iov_len; i++) {
|
||||||
|
if (data[i] < 0x20
|
||||||
|
|| data[i] == HDLC_FLAG_SEQUENCE
|
||||||
|
|| data[i] == HDLC_CONTROL_ESCAPE) {
|
||||||
|
// needs escape
|
||||||
|
packet[packetSize++] = HDLC_CONTROL_ESCAPE;
|
||||||
|
packet[packetSize++] = data[i] ^ 0x20;
|
||||||
|
} else
|
||||||
|
packet[packetSize++] = data[i];
|
||||||
|
}
|
||||||
|
// next io vector
|
||||||
|
ioVector++;
|
||||||
|
}
|
||||||
|
// mark frame end
|
||||||
|
packet[packetSize++] = HDLC_FLAG_SEQUENCE;
|
||||||
|
|
||||||
|
// send HDLC packet
|
||||||
|
|
||||||
|
bytesWritten = write(device->fd, packet, packetSize);
|
||||||
|
if (bytesWritten < 0) {
|
||||||
|
device->stats.send.errors++;
|
||||||
|
status = errno;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
device->stats.send.packets++;
|
||||||
|
device->stats.send.bytes += bytesWritten;
|
||||||
|
status = B_OK;
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
err:
|
||||||
|
device->stats.send.errors++;
|
||||||
|
|
||||||
|
done:
|
||||||
|
free(ioVectors);
|
||||||
|
free(packet);
|
||||||
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user