/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/* vim:set expandtab ts=4 shiftwidth=4: */
/* 
 * Copyright (C) 2008 Sun Microsystems, Inc. All rights reserved.
 * Use is subject to license terms.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General
 * Public License along with this library; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
 * Boston, MA 02111-1307, USA.
 *
 * Authors: Lin Ma <lin.ma@sun.com>
 */

#include "config.h"
#include <glib.h>
#include "fen-data.h"
#include "fen-missing.h"

G_LOCK_EXTERN (fen_lock);
#define SCAN_MISSING_INTERVAL 4000	/* in milliseconds */

#ifdef GIO_COMPILATION
#define FM_W if (fm_debug_enabled) g_warning
gboolean fm_debug_enabled = FALSE;
#else
#include "gam_error.h"
#define FM_W(...) GAM_DEBUG(DEBUG_INFO, __VA_ARGS__)
#endif

/* global data structure for scan missing files */
static GList *missing_list = NULL;
static guint scan_missing_source_id = 0;

static gboolean scan_missing_list (gpointer data);

static gboolean
scan_missing_list (gpointer data)
{
    GList *existing_list = NULL;
    GList *idx = NULL;
    fdata *f;
    gboolean ret = TRUE;

    G_LOCK (fen_lock);
    
    for (idx = missing_list; idx; idx = idx->next) {
        f = (fdata*)idx->data;
        
        if (port_add (&f->fobj, &f->len, f)) {
            /* TODO - emit CREATE event */
            fdata_emit_events (f, FN_EVENT_CREATED);
            existing_list = g_list_prepend (existing_list, idx);
        }
    }
    
    for (idx = existing_list; idx; idx = idx->next) {
        missing_list = g_list_remove_link (missing_list, (GList *)idx->data);
        g_list_free_1 ((GList *)idx->data);
    }
    g_list_free (existing_list);

    if (missing_list == NULL) {
        scan_missing_source_id = 0;
        ret = FALSE;
    }

    G_UNLOCK (fen_lock);
    return ret;
}

/**
 * missing_add
 *
 * Unsafe, need lock fen_lock.
 */
void
missing_add (fdata *f)
{
    GList *idx;
    
    g_assert (!is_ported (f));

    if (g_list_find (missing_list, f) != NULL) {
        FM_W ("%s is ALREADY added %s\n", __func__, FN_NAME(f));
        return;
    }
    FM_W ("%s is added %s\n", __func__, FN_NAME(f));
    
    missing_list = g_list_prepend (missing_list, f);
    
    /* if doesn't scan, then start */
    if (scan_missing_source_id == 0) {
        scan_missing_source_id = g_timeout_add (SCAN_MISSING_INTERVAL,
          scan_missing_list,
          NULL);
        g_assert (scan_missing_source_id > 0);
    }
}

/**
 * missing_remove
 *
 * Unsafe, need lock fen_lock.
 */
void
missing_remove (fdata *f)
{
    FM_W ("%s %s\n", __func__, FN_NAME(f));
    missing_list = g_list_remove (missing_list, f);
}