2007-03-26 16:49:12 -04:00
|
|
|
/* -*- c-basic-offset: 8 -*-
|
|
|
|
*
|
|
|
|
* raw1394-iso.c -- Emulation of the raw1394 rawiso API on the juju stack
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 Kristian Hoegsberg <krh@bitplanet.net>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
2007-03-28 22:55:21 -04:00
|
|
|
#include <stdio.h>
|
2007-03-26 16:49:12 -04:00
|
|
|
#include <sys/epoll.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
|
|
|
|
#include "juju.h"
|
|
|
|
|
|
|
|
static int
|
2007-04-03 13:31:36 -04:00
|
|
|
queue_xmit_packets(raw1394handle_t handle)
|
2007-03-26 16:49:12 -04:00
|
|
|
{
|
|
|
|
struct fw_cdev_iso_packet *p = handle->iso.packets;
|
2007-04-03 13:31:36 -04:00
|
|
|
struct fw_cdev_queue_iso queue_iso;
|
2007-03-26 16:49:12 -04:00
|
|
|
enum raw1394_iso_disposition d;
|
|
|
|
unsigned int len, dropped;
|
2007-04-03 21:00:06 -04:00
|
|
|
unsigned char tag, sy, *first_payload;
|
2007-04-03 13:31:36 -04:00
|
|
|
int cycle, i;
|
2007-03-26 16:49:12 -04:00
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
first_payload = handle->iso.head;
|
2007-03-26 16:49:12 -04:00
|
|
|
for (i = 0; i < handle->iso.irq_interval; i++) {
|
|
|
|
cycle = -1;
|
|
|
|
dropped = 0;
|
2007-04-03 21:00:06 -04:00
|
|
|
|
|
|
|
if (handle->iso.head + handle->iso.max_packet_size >
|
|
|
|
handle->iso.buffer_end) {
|
|
|
|
handle->iso.head = handle->iso.buffer;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
d = handle->iso.xmit_handler(handle, handle->iso.head,
|
2007-03-26 16:49:12 -04:00
|
|
|
&len, &tag, &sy, cycle, dropped);
|
|
|
|
/* FIXME: handle the different dispositions. */
|
|
|
|
|
|
|
|
p->payload_length = len;
|
2007-04-03 13:31:36 -04:00
|
|
|
p->interrupt =
|
|
|
|
handle->iso.packet_phase == handle->iso.irq_interval - 1;
|
2007-03-26 16:49:12 -04:00
|
|
|
p->skip = 0;
|
|
|
|
p->tag = tag;
|
|
|
|
p->sy = sy;
|
|
|
|
p->header_length = 0;
|
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
handle->iso.head += len;
|
2007-04-02 17:33:07 -04:00
|
|
|
handle->iso.packet_count++;
|
2007-03-28 22:55:21 -04:00
|
|
|
handle->iso.packet_phase++;
|
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
if (handle->iso.packet_phase == handle->iso.irq_interval)
|
|
|
|
handle->iso.packet_phase = 0;
|
2007-03-26 16:49:12 -04:00
|
|
|
}
|
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
queue_iso.packets = ptr_to_u64(handle->iso.packets);
|
|
|
|
queue_iso.size = i * sizeof handle->iso.packets[0];
|
2007-04-03 21:00:06 -04:00
|
|
|
queue_iso.data = ptr_to_u64(first_payload);
|
2007-04-03 13:31:36 -04:00
|
|
|
|
|
|
|
len = ioctl(handle->iso.fd, FW_CDEV_IOC_QUEUE_ISO, &queue_iso);
|
|
|
|
if (len < 0)
|
|
|
|
return -1;
|
2007-03-26 16:49:12 -04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
flush_xmit_packets(raw1394handle_t handle, int limit)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
if (handle->iso.xmit_handler == NULL)
|
|
|
|
return 0;
|
2007-03-28 22:55:21 -04:00
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
if (limit < handle->iso.irq_interval)
|
|
|
|
limit = handle->iso.irq_interval;
|
|
|
|
|
|
|
|
while (handle->iso.packet_count + handle->iso.irq_interval <= limit)
|
|
|
|
queue_xmit_packets(handle);
|
2007-03-26 16:49:12 -04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_xmit_start(raw1394handle_t handle, int start_on_cycle,
|
|
|
|
int prebuffer_packets)
|
|
|
|
{
|
|
|
|
struct fw_cdev_start_iso start_iso;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
if (prebuffer_packets == -1)
|
|
|
|
prebuffer_packets = handle->iso.irq_interval;
|
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
handle->iso.prebuffer = prebuffer_packets;
|
|
|
|
handle->iso.start_on_cycle = start_on_cycle;
|
|
|
|
|
2007-03-26 16:49:12 -04:00
|
|
|
flush_xmit_packets(handle, prebuffer_packets);
|
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
if (handle->iso.prebuffer <= handle->iso.packet_count) {
|
|
|
|
start_iso.cycle = start_on_cycle;
|
2007-03-26 16:49:12 -04:00
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
retval = ioctl(handle->iso.fd,
|
|
|
|
FW_CDEV_IOC_START_ISO, &start_iso);
|
|
|
|
if (retval < 0)
|
|
|
|
return retval;
|
|
|
|
}
|
2007-03-26 16:49:12 -04:00
|
|
|
|
|
|
|
return flush_xmit_packets(handle, handle->iso.buf_packets);
|
|
|
|
}
|
2007-03-28 22:55:21 -04:00
|
|
|
|
|
|
|
static int
|
|
|
|
queue_recv_packets(raw1394handle_t handle)
|
|
|
|
{
|
|
|
|
struct fw_cdev_queue_iso queue_iso;
|
|
|
|
struct fw_cdev_iso_packet *p = handle->iso.packets;
|
|
|
|
unsigned int len;
|
2007-04-03 21:00:06 -04:00
|
|
|
unsigned char *first_payload;
|
2007-04-02 17:33:07 -04:00
|
|
|
int i;
|
2007-03-28 22:55:21 -04:00
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
first_payload = handle->iso.head;
|
2007-03-28 22:55:21 -04:00
|
|
|
for (i = 0; i < handle->iso.irq_interval; i++, p++) {
|
2007-04-03 21:00:06 -04:00
|
|
|
if (handle->iso.head + handle->iso.max_packet_size >
|
|
|
|
handle->iso.buffer_end) {
|
|
|
|
handle->iso.head = handle->iso.buffer;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-03-28 22:55:21 -04:00
|
|
|
p->payload_length = handle->iso.max_packet_size;
|
|
|
|
p->interrupt = handle->iso.packet_phase == handle->iso.irq_interval - 1;
|
|
|
|
p->skip = 0;
|
|
|
|
p->tag = 0;
|
|
|
|
p->sy = 0;
|
|
|
|
p->header_length = 4;
|
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
handle->iso.head += handle->iso.max_packet_size;
|
2007-04-02 17:33:07 -04:00
|
|
|
handle->iso.packet_count++;
|
2007-03-28 22:55:21 -04:00
|
|
|
handle->iso.packet_phase++;
|
|
|
|
|
|
|
|
if (handle->iso.packet_phase == handle->iso.irq_interval)
|
|
|
|
handle->iso.packet_phase = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
queue_iso.packets = ptr_to_u64(handle->iso.packets);
|
2007-04-02 17:33:07 -04:00
|
|
|
queue_iso.size = i * sizeof handle->iso.packets[0];
|
2007-04-03 21:00:06 -04:00
|
|
|
queue_iso.data = ptr_to_u64(first_payload);
|
2007-03-28 22:55:21 -04:00
|
|
|
|
|
|
|
len = ioctl(handle->iso.fd, FW_CDEV_IOC_QUEUE_ISO, &queue_iso);
|
|
|
|
if (len < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2007-03-26 16:49:12 -04:00
|
|
|
|
|
|
|
static int
|
2007-03-28 22:55:21 -04:00
|
|
|
flush_recv_packets(raw1394handle_t handle,
|
|
|
|
struct fw_cdev_event_iso_interrupt *interrupt)
|
2007-03-26 16:49:12 -04:00
|
|
|
{
|
|
|
|
enum raw1394_iso_disposition d;
|
|
|
|
quadlet_t header, *p, *end;
|
|
|
|
unsigned int len, cycle, dropped;
|
|
|
|
unsigned char channel, tag, sy;
|
|
|
|
unsigned char *data;
|
|
|
|
|
|
|
|
p = interrupt->header;
|
|
|
|
end = (void *) interrupt->header + interrupt->header_length;
|
|
|
|
cycle = interrupt->cycle;
|
2007-03-28 22:55:21 -04:00
|
|
|
dropped = 0;
|
2007-03-26 16:49:12 -04:00
|
|
|
|
|
|
|
while (p < end) {
|
|
|
|
header = be32_to_cpu(*p++);
|
2007-03-31 19:45:39 -04:00
|
|
|
len = header >> 16;
|
|
|
|
tag = (header >> 14) & 0x3;
|
|
|
|
channel = (header >> 8) & 0x3f;
|
|
|
|
sy = header & 0x0f;
|
2007-03-26 16:49:12 -04:00
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
d = handle->iso.recv_handler(handle, handle->iso.tail, len,
|
|
|
|
channel, tag, sy, cycle, dropped);
|
2007-03-26 16:49:12 -04:00
|
|
|
cycle++;
|
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
handle->iso.tail += handle->iso.max_packet_size;
|
2007-04-02 17:33:07 -04:00
|
|
|
handle->iso.packet_count--;
|
2007-04-03 21:00:06 -04:00
|
|
|
if (handle->iso.tail == handle->iso.buffer_end)
|
|
|
|
handle->iso.tail = handle->iso.buffer;
|
2007-04-02 17:33:07 -04:00
|
|
|
}
|
2007-03-31 19:45:39 -04:00
|
|
|
|
2007-04-02 17:33:07 -04:00
|
|
|
while (handle->iso.packet_count + handle->iso.irq_interval <=
|
|
|
|
handle->iso.buf_packets)
|
|
|
|
queue_recv_packets(handle);
|
2007-03-28 22:55:21 -04:00
|
|
|
|
2007-03-26 16:49:12 -04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_recv_start(raw1394handle_t handle, int start_on_cycle,
|
|
|
|
int tag_mask, int sync)
|
|
|
|
{
|
|
|
|
struct fw_cdev_start_iso start_iso;
|
|
|
|
|
2007-04-02 17:33:07 -04:00
|
|
|
while (handle->iso.packet_count + handle->iso.irq_interval <=
|
2007-03-28 22:55:21 -04:00
|
|
|
handle->iso.buf_packets)
|
|
|
|
queue_recv_packets(handle);
|
|
|
|
|
2007-03-26 16:49:12 -04:00
|
|
|
start_iso.cycle = start_on_cycle;
|
|
|
|
start_iso.tags =
|
|
|
|
tag_mask == -1 ? FW_CDEV_ISO_CONTEXT_MATCH_ALL_TAGS : tag_mask;
|
|
|
|
/* sync is documented as 'not used' */
|
|
|
|
start_iso.sync = 0;
|
|
|
|
|
|
|
|
return ioctl(handle->iso.fd, FW_CDEV_IOC_START_ISO, &start_iso);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int handle_iso_event(raw1394handle_t handle,
|
|
|
|
struct epoll_closure *closure, __uint32_t events)
|
|
|
|
{
|
|
|
|
struct fw_cdev_event_iso_interrupt *interrupt;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
len = read(handle->iso.fd, handle->buffer, sizeof handle->buffer);
|
|
|
|
if (len < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
interrupt = (struct fw_cdev_event_iso_interrupt *) handle->buffer;
|
2007-03-28 22:55:21 -04:00
|
|
|
if (interrupt->type != FW_CDEV_EVENT_ISO_INTERRUPT)
|
2007-03-26 16:49:12 -04:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
switch (handle->iso.type) {
|
|
|
|
case FW_CDEV_ISO_CONTEXT_TRANSMIT:
|
2007-04-03 13:31:36 -04:00
|
|
|
handle->iso.packet_count -= handle->iso.irq_interval;
|
2007-03-26 16:49:12 -04:00
|
|
|
return flush_xmit_packets(handle, handle->iso.buf_packets);
|
|
|
|
case FW_CDEV_ISO_CONTEXT_RECEIVE:
|
2007-03-28 22:55:21 -04:00
|
|
|
return flush_recv_packets(handle, interrupt);
|
2007-03-26 16:49:12 -04:00
|
|
|
default:
|
|
|
|
/* Doesn't happen. */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_xmit_write(raw1394handle_t handle, unsigned char *data,
|
|
|
|
unsigned int len, unsigned char tag,
|
|
|
|
unsigned char sy)
|
|
|
|
{
|
2007-04-03 13:31:36 -04:00
|
|
|
struct fw_cdev_queue_iso queue_iso;
|
|
|
|
struct fw_cdev_start_iso start_iso;
|
|
|
|
struct fw_cdev_iso_packet *p;
|
2007-04-03 21:00:06 -04:00
|
|
|
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
__u32 *p = (__u32 *) data;
|
|
|
|
|
|
|
|
for (i = 0; i < 10; i++)
|
|
|
|
fprintf(stderr, "0x%08x ", p[i]);
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
2007-03-26 16:49:12 -04:00
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
if (len > handle->iso.max_packet_size) {
|
|
|
|
errno = EINVAL;
|
|
|
|
return -1;
|
|
|
|
}
|
2007-03-26 16:49:12 -04:00
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
/* Block until we have space for another packet. */
|
|
|
|
while (handle->iso.packet_count + handle->iso.irq_interval >
|
|
|
|
handle->iso.buf_packets)
|
2007-04-03 13:31:36 -04:00
|
|
|
raw1394_loop_iterate(handle);
|
|
|
|
|
|
|
|
p = &handle->iso.packets[handle->iso.packet_header_index];
|
|
|
|
p->payload_length = len;
|
|
|
|
p->interrupt =
|
|
|
|
handle->iso.packet_phase == handle->iso.irq_interval - 1;
|
|
|
|
p->skip = 0;
|
|
|
|
p->tag = tag;
|
|
|
|
p->sy = sy;
|
|
|
|
p->header_length = 0;
|
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
memcpy(handle->iso.head, data, len);
|
2007-04-03 13:31:36 -04:00
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
handle->iso.head += len;
|
2007-04-03 13:31:36 -04:00
|
|
|
handle->iso.packet_count++;
|
2007-03-26 16:49:12 -04:00
|
|
|
handle->iso.packet_phase++;
|
2007-04-03 13:31:36 -04:00
|
|
|
handle->iso.packet_header_index++;
|
|
|
|
|
2007-03-26 16:49:12 -04:00
|
|
|
if (handle->iso.packet_phase == handle->iso.irq_interval)
|
|
|
|
handle->iso.packet_phase = 0;
|
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
if (handle->iso.head + handle->iso.max_packet_size > handle->iso.buffer_end)
|
|
|
|
handle->iso.head = handle->iso.buffer;
|
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
/* Queue the packets in the kernel if we filled up the packets
|
|
|
|
* array or wrapped the payload buffer. */
|
|
|
|
if (handle->iso.packet_header_index == handle->iso.irq_interval ||
|
2007-04-03 21:00:06 -04:00
|
|
|
handle->iso.head == handle->iso.buffer) {
|
2007-04-03 13:31:36 -04:00
|
|
|
queue_iso.packets = ptr_to_u64(handle->iso.packets);
|
|
|
|
queue_iso.size = handle->iso.packet_header_index * sizeof handle->iso.packets[0];
|
2007-04-03 21:00:06 -04:00
|
|
|
queue_iso.data = ptr_to_u64(handle->iso.first_payload);
|
2007-04-03 13:31:36 -04:00
|
|
|
handle->iso.packet_header_index = 0;
|
2007-04-03 21:00:06 -04:00
|
|
|
handle->iso.first_payload = handle->iso.head;
|
2007-03-26 16:49:12 -04:00
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
len = ioctl(handle->iso.fd, FW_CDEV_IOC_QUEUE_ISO, &queue_iso);
|
|
|
|
if (len < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
2007-03-26 16:49:12 -04:00
|
|
|
|
2007-04-03 13:31:36 -04:00
|
|
|
/* Start the streaming if it's not already running and if
|
|
|
|
* we've buffered up enough packets. */
|
|
|
|
if (handle->iso.prebuffer > 0 &&
|
|
|
|
handle->iso.packet_count >= handle->iso.prebuffer) {
|
|
|
|
/* Set this to 0 to indicate that we're running. */
|
|
|
|
handle->iso.prebuffer = 0;
|
|
|
|
start_iso.cycle = handle->iso.start_on_cycle;
|
|
|
|
|
|
|
|
len = ioctl(handle->iso.fd,
|
|
|
|
FW_CDEV_IOC_START_ISO, &start_iso);
|
|
|
|
if (len < 0)
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2007-03-26 16:49:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_xmit_sync(raw1394handle_t handle)
|
|
|
|
{
|
|
|
|
/* FIXME: queue a skip packet and wait for that interrupt. */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_recv_flush(raw1394handle_t handle)
|
|
|
|
{
|
|
|
|
/* FIXME: huh, we'll need kernel support here... */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-03-28 22:55:21 -04:00
|
|
|
static int
|
|
|
|
iso_init(raw1394handle_t handle, int type,
|
|
|
|
raw1394_iso_xmit_handler_t xmit_handler,
|
|
|
|
raw1394_iso_recv_handler_t recv_handler,
|
|
|
|
unsigned int buf_packets,
|
|
|
|
unsigned int max_packet_size,
|
|
|
|
unsigned char channel,
|
|
|
|
enum raw1394_iso_speed speed,
|
|
|
|
int irq_interval)
|
2007-03-26 16:49:12 -04:00
|
|
|
{
|
|
|
|
struct fw_cdev_create_iso_context create;
|
|
|
|
struct epoll_event ep;
|
2007-03-28 22:55:21 -04:00
|
|
|
int retval, prot;
|
2007-03-26 16:49:12 -04:00
|
|
|
|
|
|
|
if (handle->iso.fd != -1) {
|
|
|
|
errno = EBUSY;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2007-03-28 22:55:21 -04:00
|
|
|
handle->iso.type = type;
|
|
|
|
if (irq_interval < 0)
|
|
|
|
handle->iso.irq_interval = 256;
|
|
|
|
else
|
|
|
|
handle->iso.irq_interval = irq_interval;
|
|
|
|
handle->iso.xmit_handler = xmit_handler;
|
|
|
|
handle->iso.recv_handler = recv_handler;
|
2007-03-26 16:49:12 -04:00
|
|
|
handle->iso.buf_packets = buf_packets;
|
|
|
|
handle->iso.max_packet_size = max_packet_size;
|
|
|
|
handle->iso.packet_phase = 0;
|
2007-04-02 17:33:07 -04:00
|
|
|
handle->iso.packet_count = 0;
|
2007-03-26 16:49:12 -04:00
|
|
|
handle->iso.packets =
|
2007-03-28 22:55:21 -04:00
|
|
|
malloc(handle->iso.irq_interval * sizeof handle->iso.packets[0]);
|
2007-03-26 16:49:12 -04:00
|
|
|
if (handle->iso.packets == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
handle->iso.fd = open(handle->local_filename, O_RDWR);
|
|
|
|
if (handle->iso.fd < 0) {
|
|
|
|
free(handle->iso.packets);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
handle->iso.closure.func = handle_iso_event;
|
|
|
|
ep.events = EPOLLIN;
|
|
|
|
ep.data.ptr = &handle->iso.closure;
|
|
|
|
if (epoll_ctl(handle->epoll_fd, EPOLL_CTL_ADD,
|
|
|
|
handle->iso.fd, &ep) < 0) {
|
|
|
|
close(handle->iso.fd);
|
|
|
|
free(handle->iso.packets);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2007-03-28 22:55:21 -04:00
|
|
|
create.type = type;
|
2007-03-26 16:49:12 -04:00
|
|
|
create.channel = channel;
|
|
|
|
create.speed = speed;
|
2007-03-28 22:55:21 -04:00
|
|
|
create.header_size = 4;
|
2007-03-26 16:49:12 -04:00
|
|
|
|
|
|
|
retval = ioctl(handle->iso.fd,
|
|
|
|
FW_CDEV_IOC_CREATE_ISO_CONTEXT, &create);
|
|
|
|
if (retval < 0) {
|
|
|
|
close(handle->iso.fd);
|
|
|
|
free(handle->iso.packets);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2007-03-28 22:55:21 -04:00
|
|
|
switch (type) {
|
|
|
|
case FW_CDEV_ISO_CONTEXT_TRANSMIT:
|
|
|
|
prot = PROT_READ | PROT_WRITE;
|
|
|
|
break;
|
|
|
|
case FW_CDEV_ISO_CONTEXT_RECEIVE:
|
|
|
|
prot = PROT_READ;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-03-26 16:49:12 -04:00
|
|
|
handle->iso.buffer =
|
|
|
|
mmap(NULL, buf_packets * max_packet_size,
|
2007-03-28 22:55:21 -04:00
|
|
|
prot, MAP_SHARED, handle->iso.fd, 0);
|
2007-03-26 16:49:12 -04:00
|
|
|
|
|
|
|
if (handle->iso.buffer == MAP_FAILED) {
|
|
|
|
close(handle->iso.fd);
|
|
|
|
free(handle->iso.packets);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2007-04-03 21:00:06 -04:00
|
|
|
handle->iso.buffer_end = handle->iso.buffer +
|
|
|
|
buf_packets * max_packet_size;
|
|
|
|
handle->iso.head = handle->iso.buffer;
|
|
|
|
handle->iso.tail = handle->iso.buffer;
|
|
|
|
handle->iso.first_payload = handle->iso.buffer;
|
|
|
|
|
2007-03-26 16:49:12 -04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-03-28 22:55:21 -04:00
|
|
|
int raw1394_iso_xmit_init(raw1394handle_t handle,
|
|
|
|
raw1394_iso_xmit_handler_t handler,
|
|
|
|
unsigned int buf_packets,
|
|
|
|
unsigned int max_packet_size,
|
|
|
|
unsigned char channel,
|
|
|
|
enum raw1394_iso_speed speed,
|
|
|
|
int irq_interval)
|
|
|
|
{
|
|
|
|
return iso_init(handle, FW_CDEV_ISO_CONTEXT_TRANSMIT,
|
|
|
|
handler, NULL, buf_packets, max_packet_size,
|
|
|
|
channel, speed, irq_interval);
|
|
|
|
}
|
|
|
|
|
2007-03-26 16:49:12 -04:00
|
|
|
int raw1394_iso_recv_init(raw1394handle_t handle,
|
|
|
|
raw1394_iso_recv_handler_t handler,
|
|
|
|
unsigned int buf_packets,
|
|
|
|
unsigned int max_packet_size,
|
|
|
|
unsigned char channel,
|
|
|
|
enum raw1394_iso_dma_recv_mode mode,
|
|
|
|
int irq_interval)
|
|
|
|
{
|
2007-03-28 22:55:21 -04:00
|
|
|
return iso_init(handle, FW_CDEV_ISO_CONTEXT_RECEIVE,
|
|
|
|
NULL, handler, buf_packets, max_packet_size,
|
|
|
|
channel, 0, irq_interval);
|
2007-03-26 16:49:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_multichannel_recv_init(raw1394handle_t handle,
|
|
|
|
raw1394_iso_recv_handler_t handler,
|
|
|
|
unsigned int buf_packets,
|
|
|
|
unsigned int max_packet_size,
|
|
|
|
int irq_interval)
|
|
|
|
{
|
|
|
|
/* FIXME: gah */
|
|
|
|
errno = ENOSYS;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_recv_listen_channel(raw1394handle_t handle,
|
|
|
|
unsigned char channel)
|
|
|
|
{
|
|
|
|
/* FIXME: multichannel */
|
|
|
|
errno = ENOSYS;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_recv_unlisten_channel(raw1394handle_t handle,
|
|
|
|
unsigned char channel)
|
|
|
|
{
|
|
|
|
/* FIXME: multichannel */
|
|
|
|
errno = ENOSYS;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int raw1394_iso_recv_set_channel_mask(raw1394handle_t handle, u_int64_t mask)
|
|
|
|
{
|
|
|
|
/* FIXME: multichannel */
|
|
|
|
errno = ENOSYS;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void raw1394_iso_stop(raw1394handle_t handle)
|
|
|
|
{
|
|
|
|
ioctl(handle->iso.fd, FW_CDEV_IOC_STOP_ISO);
|
2007-04-03 21:00:06 -04:00
|
|
|
handle->iso.head = handle->iso.buffer;
|
|
|
|
handle->iso.tail = handle->iso.buffer;
|
|
|
|
handle->iso.first_payload = handle->iso.buffer;
|
2007-04-02 17:33:07 -04:00
|
|
|
handle->iso.packet_phase = 0;
|
|
|
|
handle->iso.packet_count = 0;
|
2007-03-26 16:49:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void raw1394_iso_shutdown(raw1394handle_t handle)
|
|
|
|
{
|
|
|
|
munmap(handle->iso.buffer,
|
|
|
|
handle->iso.buf_packets * handle->iso.max_packet_size);
|
|
|
|
close(handle->iso.fd);
|
|
|
|
free(handle->iso.packets);
|
|
|
|
}
|