/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/***************************************************************************
 *            dv1394.cc
 *
 *  Tue Apr 19 12:10:34 CEST 2005
 *  Copyright  2005 Bent Bisballe
 *  deva@aasimon.org
 ****************************************************************************/

/*
 *  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 Library 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.
 */
/*
 * $Id$
 */
/*
 * $Log$
 * Revision 1.6  2005/05/01 09:56:26  deva
 * Added Id and Log tags to all files
 *
 */
#include <config.h>
#include "dv1394.h"

#include "dv.h"


#include <stdlib.h>
#include <memory.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>



/**
 * Callback function for the firewire interface.
 */
static int raw_reader( raw1394handle_t handle, int channel, size_t length, quadlet_t *data )
{
	static char *framedata = NULL;

  // Only process packets with reasonable length.
	if ( length > 16 )
	{
		unsigned char * p = ( unsigned char* ) & data[ 3 ];
		int section_type = p[ 0 ] >> 5;           // section type is in bits 5 - 7
		int dif_sequence = p[ 1 ] >> 4;           // dif sequence number is in bits 4 - 7
		int dif_block = p[ 2 ];

		if ( section_type == 0 && dif_sequence == 0 )
		{
			if ( framedata != NULL )
			{
				raw1394_set_userdata(handle, (void *)framedata);
				framedata = NULL;
			}
		}

		if(!framedata)
		{
			framedata = (char *)malloc(DVPACKAGE_SIZE); // dvframe.h
			if(!framedata)
			{
				// We're fucked
        fprintf(stderr, "Framedata allocation error: %s.\n", strerror( errno ) ); fflush(stderr);
				exit(1);
			}
		}

		switch ( section_type )
		{
		case 0:    // 1 Header block
			// p[3] |= 0x80; // hack to force PAL data
			memcpy( framedata + dif_sequence * 150 * 80, p, 480 );
			break;

		case 1:    // 2 Subcode blocks
			memcpy( framedata + dif_sequence * 150 * 80 + ( 1 + dif_block ) * 80, p, 480 );
			break;

		case 2:    // 3 VAUX blocks
			memcpy( framedata + dif_sequence * 150 * 80 + ( 3 + dif_block ) * 80, p, 480 );
			break;

		case 3:    // 9 Audio blocks interleaved with video
			memcpy( framedata + dif_sequence * 150 * 80 + ( 6 + dif_block * 16 ) * 80, p, 480 );
			break;

		case 4:    // 135 Video blocks interleaved with audio
			memcpy( framedata + dif_sequence * 150 * 80 + ( 7 + ( dif_block / 15 ) + dif_block ) * 80, p, 480 );
			break;

		default:   // we can't handle any other data
			break;
		}
	}
	return 0;
}

dv1394::dv1394(Error *e, int port, int channel)
{
  errobj = e;
	int n_ports;
	struct raw1394_portinfo pinf[ 16 ];

  // Get handle to firewire channels
	handle = raw1394_new_handle();
	if(!handle) {
    errobj->pushError("raw1394 - failed to get handle: %s.", strerror( errno ) );
    return;
	}

  // how many adapters are hooked in?
	if((n_ports = raw1394_get_port_info(handle, pinf, 16)) < 0 ) {
    errobj->pushError("raw1394 - failed to get port info: %s.", strerror( errno ) );
    raw1394_destroy_handle(handle);
    handle = NULL;
    return;
	}

	// Tell raw1394 which host adapter to use
	if(raw1394_set_port(handle, port) < 0 ) {
    errobj->pushError("raw1394 - failed to set port: %s.", strerror( errno ) );
    raw1394_destroy_handle(handle);
    handle = NULL;
    return;
	}

	raw1394_set_iso_handler( handle, channel, raw_reader);
	raw1394_set_userdata( handle, ( void* ) NULL);
  raw1394_start_iso_rcv( handle, channel);

}

dv1394::~dv1394()
{
  // Close firewire connection.
  if(handle) raw1394_destroy_handle(handle);
}

unsigned char *dv1394::readFrame()
{
  // Firewire port not correctly opened.
  if(!handle) return NULL;

  unsigned char *ptr;
  while(1) {
    raw1394_loop_iterate(handle);
    ptr = (unsigned char *)raw1394_get_userdata(handle);
    if(ptr) {
      raw1394_set_userdata(handle, NULL);
      break;
    }
  }
  return ptr;
}