/*****************************************************************************
 * vpar_motion.c : motion vectors parsing
 *****************************************************************************
 * Copyright (C) 1999, 2000 VideoLAN
 *
 * Authors:
 *
 * 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, USA.
 *****************************************************************************/

/*****************************************************************************
 * Preamble
 *****************************************************************************/
#include "defs.h"

#include <stdlib.h>                                                /* free() */
#include <sys/types.h>                        /* on BSD, uio.h needs types.h */
#include <sys/uio.h>                                            /* "input.h" */

#include "config.h"
#include "common.h"
#include "threads.h"
#include "mtime.h"
#include "plugins.h"

#include "intf_msg.h"

#include "input.h"
#include "decoder_fifo.h"
#include "video.h"
#include "video_output.h"

#include "vdec_idct.h"
#include "video_decoder.h"
#include "vdec_motion.h"

#include "vpar_blocks.h"
#include "vpar_headers.h"
#include "vpar_synchro.h"
#include "video_parser.h"

#define MAX_COUNT 3

/*
 * Local prototypes
 */

#ifdef SAM_SYNCHRO

/*****************************************************************************
 * vpar_SynchroUpdateStructures : Update the synchro structures
 *****************************************************************************/
void vpar_SynchroUpdateStructures( vpar_thread_t * p_vpar,
                                   int i_coding_type, boolean_t b_kept )
{
    double          i_can_display;
    mtime_t         i_pts;
    pes_packet_t *  p_pes = p_vpar->bit_stream.p_decoder_fifo->buffer[
                               p_vpar->bit_stream.p_decoder_fifo->i_start ];

    /* try to guess the current DTS and PTS */
    if( p_pes->b_has_pts )
    {
        i_pts = p_pes->i_pts;

        /* if the image is I type, then the presentation timestamp is
         * the PTS of the PES. Otherwise, we calculate it with the
         * theorical framerate value */
        if( i_coding_type == I_CODING_TYPE )
        {
            p_vpar->synchro.i_last_pts = p_pes->i_pts;
        }
        else
        {
            p_vpar->synchro.i_last_pts += p_vpar->synchro.i_theorical_delay;
        }

        p_pes->b_has_pts = 0;
    }
    else
    {
        p_vpar->synchro.i_last_pts += p_vpar->synchro.i_theorical_delay;
        i_pts = p_vpar->synchro.i_last_pts;
    }

    /* update structures */
    switch(i_coding_type)
    {
        case P_CODING_TYPE:

            p_vpar->synchro.i_P_seen++;
            if( b_kept ) p_vpar->synchro.i_P_kept++;
            break;

        case B_CODING_TYPE:
            p_vpar->synchro.i_B_seen++;
            if( b_kept ) p_vpar->synchro.i_B_kept++;
            break;

        case I_CODING_TYPE:

            /* update the last I PTS we have, we need it to
             * calculate the theorical framerate */
            if (i_pts != p_vpar->synchro.i_last_seen_I_pts)
            {
                if ( p_vpar->synchro.i_last_seen_I_pts )
                {
                    p_vpar->synchro.i_theorical_delay =
                            ( i_pts - p_vpar->synchro.i_last_seen_I_pts )
                          / ( 1 + p_vpar->synchro.i_B_seen
                                + p_vpar->synchro.i_P_seen);
                }
                p_vpar->synchro.i_last_seen_I_pts = i_pts;
            }

            /* now we calculated all statistics, it's time to
             * decide what we have the time to display */
            i_can_display = (float)(i_pts - p_vpar->synchro.i_last_kept_I_pts)
                                / p_vpar->synchro.i_delay;

            p_vpar->synchro.b_all_I = 0;
            p_vpar->synchro.b_all_B = 0;
            p_vpar->synchro.b_all_P = 0;
            p_vpar->synchro.displayable_p = 0;
            p_vpar->synchro.displayable_b = 0;

            if( ( p_vpar->synchro.b_all_I = ( i_can_display > 1 ) ) )
            {
                i_can_display -= 1;

                if( !( p_vpar->synchro.b_all_P
                        = ( i_can_display > p_vpar->synchro.i_P_seen ) ) )
                {
                    p_vpar->synchro.displayable_p = i_can_display;
                }
                else
                {
                    i_can_display -= p_vpar->synchro.i_P_seen;

                    if( !( p_vpar->synchro.b_all_B
                            = ( i_can_display > p_vpar->synchro.i_B_seen ) ) )
                    {
                        p_vpar->synchro.displayable_b = i_can_display;
                    }
                }
            }

#if 1
            if( p_vpar->synchro.b_all_I )
                intf_ErrMsg( " I: 1/1  " );
            if( p_vpar->synchro.b_all_P )
                intf_ErrMsg( "P: %i/%i  ", p_vpar->synchro.i_P_seen,
                                           p_vpar->synchro.i_P_seen );
            else if( p_vpar->synchro.displayable_p > 0 )
                intf_ErrMsg( "P: %.2f/%i  ", p_vpar->synchro.displayable_p,
                                             p_vpar->synchro.i_P_seen );
            if( p_vpar->synchro.b_all_B )
                intf_ErrMsg( "B: %i/%i", p_vpar->synchro.i_B_seen,
                                         p_vpar->synchro.i_B_seen );
            else if( p_vpar->synchro.displayable_b > 0 )
                intf_ErrMsg( "B: %.2f/%i", p_vpar->synchro.displayable_b,
                                           p_vpar->synchro.i_B_seen );
            intf_ErrMsg( "                    
" );
#endif
            p_vpar->synchro.i_P_seen = 0;
            p_vpar->synchro.i_B_seen = 0;

            /* update some values */
            if( b_kept )
            {
                p_vpar->synchro.i_last_kept_I_pts = i_pts;
                p_vpar->synchro.i_P_kept = 0;
                p_vpar->synchro.i_B_kept = 0;
            }

            break;
    }
}

/*****************************************************************************
 * vpar_SynchroChoose : Decide whether we will decode a picture or not
 *****************************************************************************/
boolean_t vpar_SynchroChoose( vpar_thread_t * p_vpar, int i_coding_type,
                              int i_structure )
{
    mtime_t i_delay = p_vpar->synchro.i_last_pts - mdate();

    //return (i_coding_type == I_CODING_TYPE);

    switch( i_coding_type )
    {
        case I_CODING_TYPE:

            //intf_ErrMsg( " I	%f	%f\nI ", 1000000.0 / p_vpar->synchro.i_theorical_delay, 1000000.0 / p_vpar->synchro.i_delay );
            return( p_vpar->synchro.b_all_I );

        case P_CODING_TYPE:

            //return(1);
            if( p_vpar->synchro.b_all_P )
            {
                //intf_ErrMsg( " p  " );
                return( 1 );
            }

            if( p_vpar->synchro.displayable_p * i_delay
                < p_vpar->synchro.i_delay )
            {
                //intf_ErrMsg( " -  " );
                return( 0 );
            }

            p_vpar->synchro.displayable_p--;
            //intf_ErrMsg( " p> " );
            return( 1 );

        case B_CODING_TYPE:

            if( p_vpar->synchro.b_all_B )
            {
                //intf_ErrMsg( "b " );
                return( 1 );
            }

            if( p_vpar->synchro.displayable_b <= 0 )
            {
                //intf_ErrMsg( "  " );
                return( 0 );
            }

            if( i_delay < 0 )
            {
                //intf_ErrMsg( "� " );
                p_vpar->synchro.displayable_b -= 0.5;
                return( 0 );
            }

            //intf_ErrMsg( "b " );
            p_vpar->synchro.displayable_b--;
            return( 1 );
    }

    return( 0 );

}

/*****************************************************************************
 * vpar_SynchroTrash : Update timers when we trash a picture
 *****************************************************************************/
void vpar_SynchroTrash( vpar_thread_t * p_vpar, int i_coding_type,
                        int i_structure )
{
    vpar_SynchroUpdateStructures (p_vpar, i_coding_type, 0);

}

/*****************************************************************************
 * vpar_SynchroDecode : Update timers when we decide to decode a picture
 *****************************************************************************/
void vpar_SynchroDecode( vpar_thread_t * p_vpar, int i_coding_type,
                            int i_structure )
{
    vpar_SynchroUpdateStructures (p_vpar, i_coding_type, 1);

    p_vpar->synchro.i_date_fifo[p_vpar->synchro.i_stop] = mdate();

    FIFO_INCREMENT( i_stop );

}

/*****************************************************************************
 * vpar_SynchroEnd : Called when the image is totally decoded
 *****************************************************************************/
void vpar_SynchroEnd( vpar_thread_t * p_vpar )
{
    if( p_vpar->synchro.i_stop != p_vpar->synchro.i_start )
    {
        mtime_t i_delay;

        i_delay = ( mdate() -
            p_vpar->synchro.i_date_fifo[p_vpar->synchro.i_start] )
              / ( (p_vpar->synchro.i_stop - p_vpar->synchro.i_start) & 0x0f );

        p_vpar->synchro.i_delay =
            ( 7 * p_vpar->synchro.i_delay + i_delay ) >> 3;

#if 0
        intf_ErrMsg( "decode %lli (mean %lli, theorical %lli)\n",
                     i_delay, p_vpar->synchro.i_delay,
                     p_vpar->synchro.i_theorical_delay );
#endif
    }
    else
    {
        intf_ErrMsg( "vpar error: critical ! fifo full\n" );
    }

    FIFO_INCREMENT( i_start );
}

/*****************************************************************************
 * vpar_SynchroDate : When an image has been decoded, ask for its date
 *****************************************************************************/
mtime_t vpar_SynchroDate( vpar_thread_t * p_vpar )
{
#if 0

    mtime_t i_displaydate = p_vpar->synchro.i_last_pts;

    static mtime_t i_delta = 0;

    intf_ErrMsg( "displaying image with delay %lli and delta %lli\n",
        i_displaydate - mdate(),
        i_displaydate - i_delta );

    intf_ErrMsg ( "theorical fps: %f - actual fps: %f \n",
        1000000.0 / p_vpar->synchro.i_theorical_delay, 1000000.0 / p_vpar->synchro.i_delay );

    i_delta = i_displaydate;

    return i_displaydate;
#else

    return p_vpar->synchro.i_last_pts;

#endif
}

#endif

#ifdef MEUUH_SYNCHRO

/* synchro a deux balles backportee du decodeur de reference. NE MARCHE PAS
AVEC LES IMAGES MONOTRAMES */

boolean_t vpar_SynchroChoose( vpar_thread_t * p_vpar, int i_coding_type,
                              int i_structure )
{
    switch (i_coding_type)
    {
    case B_CODING_TYPE:
        if ((p_vpar->synchro.kludge_level <= p_vpar->synchro.kludge_nbp))
        {
            p_vpar->synchro.kludge_b++;
            return( 0 );
        }
        if (p_vpar->synchro.kludge_b %
             (p_vpar->synchro.kludge_nbb /
                (p_vpar->synchro.kludge_level - p_vpar->synchro.kludge_nbp)))
        {
            p_vpar->synchro.kludge_b++;
            return( 0 );
        }
        p_vpar->synchro.kludge_b++;
        return( 1 );

    case P_CODING_TYPE:
        if (p_vpar->synchro.kludge_p++ >= p_vpar->synchro.kludge_level)
        {
            return( 0 );
        }
        return( 1 );

    default:
        return( 1 );
    }
}

void vpar_SynchroTrash( vpar_thread_t * p_vpar, int i_coding_type,
                        int i_structure )
{
    if (DECODER_FIFO_START(p_vpar->fifo)->b_has_pts && i_coding_type == I_CODING_TYPE)
    {
        p_vpar->synchro.kludge_nbframes = 0;
        p_vpar->synchro.kludge_date = DECODER_FIFO_START(p_vpar->fifo)->i_pts;
    }
    else
        p_vpar->synchro.kludge_nbframes++;
    DECODER_FIFO_START(p_vpar->fifo)->b_has_pts = 0;
}

void vpar_SynchroDecode( vpar_thread_t * p_vpar, int i_coding_type,
                            int i_structure )
{
    if (DECODER_FIFO_START(p_vpar->fifo)->b_has_pts && i_coding_type == I_CODING_TYPE)
    {
        p_vpar->synchro.kludge_nbframes = 0;
        p_vpar->synchro.kludge_date = DECODER_FIFO_START(p_vpar->fifo)->i_pts;
        DECODER_FIFO_START(p_vpar->fifo)->b_has_pts = 0;
    }
    else
        p_vpar->synchro.kludge_nbframes++;
}

mtime_t vpar_SynchroDate( vpar_thread_t * p_vpar )
{
    return( p_vpar->synchro.kludge_date
            + p_vpar->synchro.kludge_nbframes*1000000/(p_vpar->sequence.r_frame_rate ) );
}

void vpar_SynchroEnd( vpar_thread_t * p_vpar )
{
}

void vpar_SynchroKludge( vpar_thread_t * p_vpar, mtime_t date )
{
    mtime_t     show_date;
    int         temp = p_vpar->synchro.kludge_level;

    p_vpar->synchro.kludge_nbp = p_vpar->synchro.kludge_p ? p_vpar->synchro.kludge_p : 5;
    p_vpar->synchro.kludge_nbb = p_vpar->synchro.kludge_b ? p_vpar->synchro.kludge_b : 6;
    show_date = date - mdate();
    p_vpar->synchro.kludge_p = 0;
    p_vpar->synchro.kludge_b = 0;

    if (show_date < (SYNC_DELAY - SYNC_TOLERATE) && show_date <= p_vpar->synchro.kludge_prevdate)
    {
        p_vpar->synchro.kludge_level--;
        if (p_vpar->synchro.kludge_level < 0)
            p_vpar->synchro.kludge_level = 0;
        else if (p_vpar->synchro.kludge_level >
                     p_vpar->synchro.kludge_nbp + p_vpar->synchro.kludge_nbb)
            p_vpar->synchro.kludge_level = p_vpar->synchro.kludge_nbp + p_vpar->synchro.kludge_nbb;
#ifdef DEBUG
        if (temp != p_vpar->synchro.kludge_level)
            intf_DbgMsg("vdec debug: Level changed from %d to %d (%Ld)\n",
                        temp, p_vpar->synchro.kludge_level, show_date );
#endif
    }
    else if (show_date > (SYNC_DELAY + SYNC_TOLERATE) && show_date >= p_vpar->synchro.kludge_prevdate)
    {
        p_vpar->synchro.kludge_level++;
        if (p_vpar->synchro.kludge_level > p_vpar->synchro.kludge_nbp + p_vpar->synchro.kludge_nbb)
            p_vpar->synchro.kludge_level = p_vpar->synchro.kludge_nbp + p_vpar->synchro.kludge_nbb;
#ifdef DEBUG
        if (temp != p_vpar->synchro.kludge_level)
            intf_DbgMsg("vdec debug: Level changed from %d to %d (%Ld)\n",
                        temp, p_vpar->synchro.kludge_level, show_date );
#endif
    }

    p_vpar->synchro.kludge_prevdate = show_date;
    if ((p_vpar->synchro.kludge_level - p_vpar->synchro.kludge_nbp) > p_vpar->synchro.kludge_nbb)
        p_vpar->synchro.kludge_level = p_vpar->synchro.kludge_nbb + p_vpar->synchro.kludge_nbp;
}

#endif


#ifdef POLUX_SYNCHRO

void vpar_SynchroSetCurrentDate( vpar_thread_t * p_vpar, int i_coding_type )
{
    pes_packet_t * p_pes =
        p_vpar->bit_stream.p_decoder_fifo->buffer[p_vpar->bit_stream.p_decoder_fifo->i_start];


    switch( i_coding_type )
    {
    case B_CODING_TYPE:
        if( p_pes->b_has_pts )
        {
            if( p_pes->i_pts < p_vpar->synchro.i_current_frame_date )
            {
                intf_ErrMsg( "vpar warning: pts_date < current_date\n" );
            }
            p_vpar->synchro.i_current_frame_date = p_pes->i_pts;
            p_pes->b_has_pts = 0;
        }
        else
        {
            p_vpar->synchro.i_current_frame_date += 1000000/(p_vpar->sequence.r_frame_rate);
        }
        break;

    default:

        if( p_vpar->synchro.i_backward_frame_date == 0 )
        {
            p_vpar->synchro.i_current_frame_date += 1000000/(p_vpar->sequence.r_frame_rate);
        }
        else
        {
            if( p_vpar->synchro.i_backward_frame_date < p_vpar->synchro.i_current_frame_date )
            {
                intf_ErrMsg( "vpar warning: backward_date < current_date (%Ld)\n",
                         p_vpar->synchro.i_backward_frame_date - p_vpar->synchro.i_current_frame_date );
            }
            p_vpar->synchro.i_current_frame_date = p_vpar->synchro.i_backward_frame_date;
            p_vpar->synchro.i_backward_frame_date = 0;
        }

        if( p_pes->b_has_pts )
        {
            p_vpar->synchro.i_backward_frame_date = p_pes->i_pts;
            p_pes->b_has_pts = 0;
        }
       break;
    }
}

boolean_t vpar_SynchroChoose( vpar_thread_t * p_vpar, int i_coding_type,
                              int i_structure )
{
    boolean_t b_result = 1;
    int i_synchro_level = p_vpar->p_vout->i_synchro_level;

    vpar_SynchroSetCurrentDate( p_vpar, i_coding_type );

    /*
     * The synchro level is updated by the video input (see SynchroLevelUpdate)
     * so we just use the synchro_level to decide which frame to trash
     */

    switch( i_coding_type )
    {
    case I_CODING_TYPE:

        p_vpar->synchro.r_p_average =
            (p_vpar->synchro.r_p_average*(SYNC_AVERAGE_COUNT-1)+p_vpar->synchro.i_p_count)/SYNC_AVERAGE_COUNT;
        p_vpar->synchro.r_b_average =
            (p_vpar->synchro.r_b_average*(SYNC_AVERAGE_COUNT-1)+p_vpar->synchro.i_b_count)/SYNC_AVERAGE_COUNT;

        p_vpar->synchro.i_p_nb = (int)(p_vpar->synchro.r_p_average+0.5);
        p_vpar->synchro.i_b_nb = (int)(p_vpar->synchro.r_b_average+0.5);

        p_vpar->synchro.i_p_count = p_vpar->synchro.i_b_count = 0;
        p_vpar->synchro.i_b_trasher = p_vpar->synchro.i_b_nb / 2;
        p_vpar->synchro.i_i_count++;
       break;

    case P_CODING_TYPE:
        p_vpar->synchro.i_p_count++;
        if( p_vpar->synchro.i_p_count > i_synchro_level )
        {
            b_result = 0;
        }
        break;

    case B_CODING_TYPE:
        p_vpar->synchro.i_b_count++;
        if( p_vpar->synchro.i_p_nb >= i_synchro_level )
        {
            /* We must trash all the B */
            b_result = 0;
        }
        else
        {
            /* We use the brensenham algorithm to decide which B to trash */
            p_vpar->synchro.i_b_trasher +=
                p_vpar->synchro.i_b_nb - (i_synchro_level-p_vpar->synchro.i_p_nb);
            if( p_vpar->synchro.i_b_trasher >= p_vpar->synchro.i_b_nb )
            {
                b_result = 0;
                p_vpar->synchro.i_b_trasher -= p_vpar->synchro.i_b_nb;
            }
        }
        break;
    }

    return( b_result );
}

void vpar_SynchroTrash( vpar_thread_t * p_vpar, int i_coding_type,
                        int i_structure )
{
    vpar_SynchroChoose( p_vpar, i_coding_type, i_structure );
}

void vpar_SynchroUpdateLevel()
{
    //vlc_mutex_lock( &level_lock );
    //vlc_mutex_unlock( &level_lock );
}

mtime_t vpar_SynchroDate( vpar_thread_t * p_vpar )
{
    return( p_vpar->synchro.i_current_frame_date );
}

/* functions with no use */

void vpar_SynchroEnd( vpar_thread_t * p_vpar )
{
}

void vpar_SynchroDecode( vpar_thread_t * p_vpar, int i_coding_type,
                            int i_structure )
{
}

#endif