/*
 * gmodules.c -- gpart module functions
 *
 * gpart (c) 1999-2001 Michail Brzitwa <mb@ichabod.han.de>
 * Guess PC-type hard disk partitions.
 *
 * gpart 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, or (at your
 * option) any later version.
 *
 * Created:   04.01.1999 <mb@ichabod.han.de>
 * Modified:  29.01.2001 <mb@ichabod.han.de>
 *            New modules: qnx & beos.
 *
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <dlfcn.h>
#include "gpart.h"


static g_module		*g_head;
static int		g_count;



g_module *g_mod_head()
{
	return (g_head);
}



int g_mod_count()
{
	return (g_count);
}



void g_mod_list()
{
	g_module	*m;

	pr(MSG,"Module\tWeight\n");
	for (m = g_head; m; m = m->m_next)
		pr(MSG,"%s\t(%3.1f)\n",m->m_name,m->m_weight);
	pr(MSG,"\n");
}



void g_mod_delete(g_module *m)
{
	if (m)
	{
		if (m->m_hd) dlclose(m->m_hd);
		if (m->m_name) free((void *)m->m_name);
		free(m);
		g_count--;
	}
}



void g_mod_deleteall()
{
	g_module	*m;

	while (g_head)
	{
		m = g_head->m_next; g_mod_delete(g_head); g_head = m;
	}
}



/*
 * set weight of module and re-insert as head. 
 */

g_module *g_mod_setweight(char *name,float weight)
{
	g_module	*m, *prev = 0;

	for (m = g_head; m; m = m->m_next)
		if (strcmp(m->m_name,name) == 0)
			break;
		else
			prev = m;
	if (m == 0)
		return (0);
	if (prev)
	{
		prev->m_next = m->m_next;
		m->m_next = g_head;
		g_head = m;
	}
	g_head->m_weight = weight;
	return (g_head);
}



g_module *g_mod_lookup(int how,char *name)
{
	g_module	*m;

	if (g_head == 0)
	{
		if (how == GM_LOOKUP)
			return (0);
		g_head = (g_module *)alloc(sizeof(g_module));
		m = g_head;
	}
	else
	{
		for (m = g_head; m->m_next; m = m->m_next)
			if (strcmp(m->m_name,name) == 0)
				return (m);
		if (how == GM_LOOKUP)
			return (0);
		m->m_next = (g_module *)alloc(sizeof(g_module));
		m = m->m_next;
	}
	if ((m->m_name = strdup(name)) == 0)
		pr(FATAL,"out of memory in strdup");
	m->m_weight = 1.0; g_count++;
	return (m);
}



/*
 * preloaded modules
 */

void g_mod_addinternals()
{
	g_module	*m;

#define GMODINS(mod)	if(!(m = g_mod_lookup(GM_INSERT,#mod))->m_hd){		\
			m->m_init=mod##_init; m->m_term=mod##_term;	\
			m->m_gfun=mod##_gfun; }

	/*
	 * If no weights are given on the command line, the order
	 * is somehow important.
	 */

	GMODINS(bsddl);
	GMODINS(lswap);
	GMODINS(qnx4);
	GMODINS(rfs);
	GMODINS(ntfs);
	GMODINS(hpfs);
	GMODINS(minix);
	GMODINS(beos);
	GMODINS(ext2);
	GMODINS(fat);
	GMODINS(s86dl);
	GMODINS(hmlvm);
	GMODINS(xfs);
}



int g_mod_addexternal(char *name)
{
	g_module	*m;
	char		buf[FILENAME_MAX];

	/*
	 * external modules are named 'gm_' name '.so', and will
	 * be searched in the standard ld.so library directories
	 * or those explicitly given by LD_LIBRARY_PATH.
	 */

	snprintf(buf,FILENAME_MAX-1,"gm_%s.so",name);
	buf[FILENAME_MAX-1] = 0;

	m = g_mod_lookup(GM_INSERT,name);
	if (m->m_hd)
		dlclose(m->m_hd);

	if ((m->m_hd = dlopen(buf,RTLD_NOW)) == 0)
		pr(FATAL,(char *)dlerror());

	snprintf(buf,FILENAME_MAX-1,"%s_init",name);
	m->m_init = (int (*)())dlsym(m->m_hd,buf);
	snprintf(buf,FILENAME_MAX-1,"%s_term",name);
	m->m_term = (int (*)())dlsym(m->m_hd,buf);
	snprintf(buf,FILENAME_MAX-1,"%s_gfun",name);
	m->m_gfun = (int (*)())dlsym(m->m_hd,buf);
	if ((m->m_gfun == 0))
		pr(FATAL,"module %s: missing vital functions",name);

	return (1);
}


syntax highlighted by Code2HTML, v. 0.9.1