2005-09-20 13:26:39 +00:00
/** \file function.c
2006-11-15 14:16:49 +00:00
Prototypes for functions for storing and retrieving function
information . These functions also take care of autoloading
functions in the $ fish_function_path . Actual function evaluation
is taken care of by the parser and to some degree the builtin
handling library .
2005-09-20 13:26:39 +00:00
*/
2006-08-11 01:18:35 +00:00
# include "config.h"
2005-09-20 13:26:39 +00:00
# include <stdlib.h>
# include <stdio.h>
2011-12-27 03:18:46 +00:00
# include <string.h>
2005-09-20 13:26:39 +00:00
# include <wchar.h>
# include <unistd.h>
# include <termios.h>
# include <signal.h>
2011-12-27 03:18:46 +00:00
# include <pthread.h>
# include <errno.h>
2012-01-14 07:44:18 +00:00
# include <map>
# include <set>
2005-09-20 13:26:39 +00:00
2006-01-20 14:27:21 +00:00
# include "wutil.h"
2006-02-28 13:17:16 +00:00
# include "fallback.h"
2005-09-20 13:26:39 +00:00
# include "util.h"
2006-02-28 13:17:16 +00:00
2005-09-20 13:26:39 +00:00
# include "function.h"
# include "proc.h"
# include "parser.h"
# include "common.h"
# include "intern.h"
2005-10-05 22:37:08 +00:00
# include "event.h"
2006-01-26 14:48:10 +00:00
# include "reader.h"
2006-02-05 13:10:35 +00:00
# include "parse_util.h"
2007-04-22 09:50:26 +00:00
# include "parser_keywords.h"
2006-02-08 09:20:05 +00:00
# include "env.h"
2006-02-08 15:29:09 +00:00
# include "expand.h"
2012-01-23 19:42:41 +00:00
# include "builtin_scripts.h"
2005-09-20 13:26:39 +00:00
2005-10-05 22:37:08 +00:00
/**
Table containing all functions
*/
2012-01-24 04:32:36 +00:00
typedef std : : map < wcstring , shared_ptr < function_info_t > > function_map_t ;
2012-01-14 07:44:18 +00:00
static function_map_t loaded_functions ;
2011-12-27 03:18:46 +00:00
/* Lock for functions */
static pthread_mutex_t functions_lock ;
2012-01-26 02:40:08 +00:00
/* Autoloader for functions */
class function_autoload_t : public autoload_t {
public :
function_autoload_t ( ) ;
virtual void command_removed ( const wcstring & cmd ) ;
} ;
static function_autoload_t function_autoloader ;
/** Constructor */
function_autoload_t : : function_autoload_t ( ) : autoload_t ( L " fish_function_path " ,
internal_function_scripts ,
sizeof internal_function_scripts / sizeof * internal_function_scripts )
{
}
/** Removes a function from our internal table, returning true if it was found and false if not */
static bool function_remove_ignore_autoload ( const wcstring & name ) ;
/** Callback when an autoloaded function is removed */
void function_autoload_t : : command_removed ( const wcstring & cmd ) {
2012-01-26 02:51:26 +00:00
function_remove_ignore_autoload ( cmd ) ;
2012-01-26 02:40:08 +00:00
}
2011-12-27 03:18:46 +00:00
/* Helper macro for vomiting */
# define VOMIT_ON_FAILURE(a) do { if (0 != (a)) { int err = errno; fprintf(stderr, "%s failed on line %d in file %s: %d (%s)\n", #a, __LINE__, __FILE__, err, strerror(err)); abort(); }} while (0)
2006-06-17 13:07:08 +00:00
/**
Kludgy flag set by the load function in order to tell function_add
that the function being defined is autoloaded . There should be a
better way to do this . . .
*/
2006-02-08 09:20:05 +00:00
static int is_autoload = 0 ;
2006-02-08 17:37:18 +00:00
/**
Make sure that if the specified function is a dynamically loaded
function , it has been fully loaded .
*/
2006-02-08 09:20:05 +00:00
static int load ( const wchar_t * name )
{
2011-12-27 05:56:23 +00:00
ASSERT_IS_MAIN_THREAD ( ) ;
2012-01-24 04:32:36 +00:00
scoped_lock lock ( functions_lock ) ;
2006-02-08 09:20:05 +00:00
int was_autoload = is_autoload ;
int res ;
2012-01-14 07:44:18 +00:00
function_map_t : : iterator iter = loaded_functions . find ( name ) ;
2012-01-24 04:32:36 +00:00
if ( iter ! = loaded_functions . end ( ) & & ! iter - > second - > is_autoload ) {
2012-01-28 22:56:13 +00:00
/* We have a non-autoload version already */
2006-02-08 09:20:05 +00:00
return 0 ;
2011-12-27 03:18:46 +00:00
}
is_autoload = 1 ;
2012-01-26 02:40:08 +00:00
res = function_autoloader . load ( name , true ) ;
2006-02-08 09:20:05 +00:00
is_autoload = was_autoload ;
return res ;
}
2006-02-08 17:37:18 +00:00
/**
Insert a list of all dynamically loaded functions into the
specified list .
*/
2012-01-14 07:44:18 +00:00
static void autoload_names ( std : : set < wcstring > & names , int get_hidden )
2006-02-08 17:37:18 +00:00
{
2012-01-10 20:51:09 +00:00
size_t i ;
2011-12-27 03:18:46 +00:00
2012-01-14 10:42:17 +00:00
const env_var_t path_var_wstr = env_get_string ( L " fish_function_path " ) ;
if ( path_var_wstr . missing ( ) )
return ;
const wchar_t * path_var = path_var_wstr . c_str ( ) ;
2011-12-27 03:18:46 +00:00
2012-01-10 20:51:09 +00:00
wcstring_list_t path_list ;
2006-02-08 17:37:18 +00:00
2012-02-10 09:37:30 +00:00
tokenize_variable_array ( path_var , path_list ) ;
2012-01-10 20:51:09 +00:00
for ( i = 0 ; i < path_list . size ( ) ; i + + )
2006-02-08 17:37:18 +00:00
{
2012-01-10 20:51:09 +00:00
const wcstring & ndir_str = path_list . at ( i ) ;
const wchar_t * ndir = ( wchar_t * ) ndir_str . c_str ( ) ;
2006-02-08 17:37:18 +00:00
DIR * dir = wopendir ( ndir ) ;
2006-02-13 21:44:16 +00:00
if ( ! dir )
continue ;
2011-12-27 03:18:46 +00:00
wcstring name ;
while ( wreaddir ( dir , name ) )
2006-02-08 17:37:18 +00:00
{
2011-12-27 03:18:46 +00:00
const wchar_t * fn = name . c_str ( ) ;
2012-01-14 07:44:18 +00:00
const wchar_t * suffix ;
2006-02-08 17:37:18 +00:00
if ( ! get_hidden & & fn [ 0 ] = = L ' _ ' )
continue ;
2011-12-27 03:18:46 +00:00
2012-01-14 07:44:18 +00:00
suffix = wcsrchr ( fn , L ' . ' ) ;
2006-02-08 17:37:18 +00:00
if ( suffix & & ( wcscmp ( suffix , L " .fish " ) = = 0 ) )
{
2012-01-14 07:44:18 +00:00
wcstring name ( fn , suffix - fn ) ;
names . insert ( name ) ;
2006-02-08 17:37:18 +00:00
}
2011-12-27 03:18:46 +00:00
}
2006-02-08 17:37:18 +00:00
closedir ( dir ) ;
}
2006-02-08 09:20:05 +00:00
}
2005-09-20 13:26:39 +00:00
void function_init ( )
{
2011-12-27 03:18:46 +00:00
pthread_mutexattr_t a ;
VOMIT_ON_FAILURE ( pthread_mutexattr_init ( & a ) ) ;
VOMIT_ON_FAILURE ( pthread_mutexattr_settype ( & a , PTHREAD_MUTEX_RECURSIVE ) ) ;
VOMIT_ON_FAILURE ( pthread_mutex_init ( & functions_lock , & a ) ) ;
VOMIT_ON_FAILURE ( pthread_mutexattr_destroy ( & a ) ) ;
2005-09-20 13:26:39 +00:00
}
2006-01-26 14:48:10 +00:00
2012-01-16 20:10:08 +00:00
void function_add ( function_data_t * data , const parser_t & parser )
2005-09-20 13:26:39 +00:00
{
2012-02-09 10:01:49 +00:00
CHECK ( ! data - > name . empty ( ) , ) ;
2007-04-22 22:10:33 +00:00
CHECK ( data - > definition , ) ;
2012-01-24 04:32:36 +00:00
scoped_lock lock ( functions_lock ) ;
2007-04-22 22:10:33 +00:00
function_remove ( data - > name ) ;
2012-01-14 07:44:18 +00:00
2012-01-24 04:32:36 +00:00
shared_ptr < function_info_t > & info = loaded_functions [ data - > name ] ;
if ( ! info ) {
info . reset ( new function_info_t ) ;
}
2012-01-14 07:44:18 +00:00
2012-01-24 04:32:36 +00:00
info - > definition_offset = parse_util_lineno ( parser . get_buffer ( ) , parser . current_block - > tok_pos ) - 1 ;
info - > definition = data - > definition ;
2007-04-16 20:06:11 +00:00
2012-02-09 07:53:23 +00:00
if ( ! data - > named_arguments . empty ( ) ) {
info - > named_arguments . insert ( info - > named_arguments . end ( ) , data - > named_arguments . begin ( ) , data - > named_arguments . end ( ) ) ;
}
2011-12-27 03:18:46 +00:00
2012-02-09 10:01:49 +00:00
if ( ! data - > description . empty ( ) )
2012-01-24 04:32:36 +00:00
info - > description = data - > description ;
info - > definition_file = intern ( reader_current_filename ( ) ) ;
info - > is_autoload = is_autoload ;
info - > shadows = data - > shadows ;
2011-12-27 03:18:46 +00:00
2012-02-08 10:34:31 +00:00
for ( size_t i = 0 ; i < data - > events . size ( ) ; i + + )
2005-10-05 22:37:08 +00:00
{
2012-02-09 03:02:25 +00:00
event_add_handler ( & data - > events . at ( i ) ) ;
2005-10-05 22:37:08 +00:00
}
2005-09-20 13:26:39 +00:00
}
2012-01-28 22:56:13 +00:00
int function_exists ( const wchar_t * cmd )
2010-09-07 17:31:05 +00:00
{
2011-12-27 03:18:46 +00:00
CHECK ( cmd , 0 ) ;
if ( parser_keywords_is_reserved ( cmd ) )
2010-09-07 17:31:05 +00:00
return 0 ;
2012-01-24 04:32:36 +00:00
scoped_lock lock ( functions_lock ) ;
2012-01-28 22:56:13 +00:00
load ( cmd ) ;
return loaded_functions . find ( cmd ) ! = loaded_functions . end ( ) ;
2011-12-27 03:18:46 +00:00
}
2010-09-18 01:51:16 +00:00
2012-01-28 22:56:13 +00:00
int function_exists_no_autoload ( const wchar_t * cmd , const env_vars & vars )
2011-12-27 03:18:46 +00:00
{
2012-01-28 22:56:13 +00:00
CHECK ( cmd , 0 ) ;
if ( parser_keywords_is_reserved ( cmd ) )
return 0 ;
scoped_lock lock ( functions_lock ) ;
return loaded_functions . find ( cmd ) ! = loaded_functions . end ( ) | | function_autoloader . can_load ( cmd , vars ) ;
2005-09-20 13:26:39 +00:00
}
2012-01-26 02:40:08 +00:00
static bool function_remove_ignore_autoload ( const wcstring & name )
2005-09-20 13:26:39 +00:00
{
2012-01-24 04:32:36 +00:00
scoped_lock lock ( functions_lock ) ;
2012-01-14 07:44:18 +00:00
bool erased = ( loaded_functions . erase ( name ) > 0 ) ;
2011-12-27 03:18:46 +00:00
2012-01-26 02:40:08 +00:00
if ( erased ) {
2012-02-09 03:02:25 +00:00
event_t ev ( EVENT_ANY ) ;
2012-01-26 02:40:08 +00:00
ev . function_name = name . c_str ( ) ;
event_remove ( & ev ) ;
2011-12-27 03:18:46 +00:00
}
2012-01-26 02:40:08 +00:00
return erased ;
2005-09-20 13:26:39 +00:00
2012-01-26 02:40:08 +00:00
}
2006-02-05 13:10:35 +00:00
2012-01-26 02:40:08 +00:00
void function_remove ( const wcstring & name )
{
if ( function_remove_ignore_autoload ( name ) )
function_autoloader . unload ( name ) ;
2012-01-24 04:32:36 +00:00
}
shared_ptr < function_info_t > function_get ( const wcstring & name )
{
scoped_lock lock ( functions_lock ) ;
function_map_t : : iterator iter = loaded_functions . find ( name ) ;
if ( iter = = loaded_functions . end ( ) ) {
return shared_ptr < function_info_t > ( ) ;
} else {
return iter - > second ;
}
2005-09-20 13:26:39 +00:00
}
2011-12-27 03:18:46 +00:00
2007-04-16 20:06:11 +00:00
const wchar_t * function_get_definition ( const wchar_t * name )
2005-09-20 13:26:39 +00:00
{
2012-01-24 04:43:39 +00:00
shared_ptr < function_info_t > func = function_get ( name ) ;
return func ? func - > definition . c_str ( ) : NULL ;
2005-09-20 13:26:39 +00:00
}
2007-04-16 20:06:11 +00:00
2012-01-14 07:44:18 +00:00
wcstring_list_t function_get_named_arguments ( const wchar_t * name )
2007-04-16 20:06:11 +00:00
{
2012-01-24 04:43:39 +00:00
shared_ptr < function_info_t > func = function_get ( name ) ;
return func ? func - > named_arguments : wcstring_list_t ( ) ;
2007-04-16 20:06:11 +00:00
}
2007-04-22 22:10:33 +00:00
int function_get_shadows ( const wchar_t * name )
{
2012-01-24 04:43:39 +00:00
shared_ptr < function_info_t > func = function_get ( name ) ;
return func ? func - > shadows : false ;
2007-04-22 22:10:33 +00:00
}
2011-12-27 03:18:46 +00:00
2007-04-16 20:06:11 +00:00
const wchar_t * function_get_desc ( const wchar_t * name )
2005-09-20 13:26:39 +00:00
{
2012-01-14 07:44:18 +00:00
/* Empty length string goes to NULL */
2012-01-24 04:43:39 +00:00
shared_ptr < function_info_t > func = function_get ( name ) ;
if ( func & & func - > description . size ( ) ) {
return _ ( func - > description . c_str ( ) ) ;
} else {
return NULL ;
}
2005-09-20 13:26:39 +00:00
}
void function_set_desc ( const wchar_t * name , const wchar_t * desc )
{
2006-06-21 00:48:36 +00:00
CHECK ( name , ) ;
CHECK ( desc , ) ;
2011-12-27 03:18:46 +00:00
2006-02-08 09:20:05 +00:00
load ( name ) ;
2012-01-24 04:43:39 +00:00
shared_ptr < function_info_t > func = function_get ( name ) ;
if ( func ) func - > description = desc ;
2005-09-20 13:26:39 +00:00
}
2011-12-27 03:18:46 +00:00
int function_copy ( const wchar_t * name , const wchar_t * new_name )
{
2012-01-14 07:44:18 +00:00
int result = 0 ;
2012-01-24 04:32:36 +00:00
scoped_lock lock ( functions_lock ) ;
2012-01-14 07:44:18 +00:00
function_map_t : : const_iterator iter = loaded_functions . find ( name ) ;
if ( iter ! = loaded_functions . end ( ) ) {
2012-01-24 04:43:39 +00:00
shared_ptr < function_info_t > & new_info = loaded_functions [ new_name ] ;
new_info . reset ( new function_info_t ( * iter - > second ) ) ;
2012-01-14 07:44:18 +00:00
2011-12-27 03:18:46 +00:00
// This new instance of the function shouldn't be tied to the def
// file of the original.
2012-01-24 04:43:39 +00:00
new_info - > definition_file = 0 ;
new_info - > is_autoload = 0 ;
2012-01-14 07:44:18 +00:00
result = 1 ;
}
return result ;
2005-09-20 13:26:39 +00:00
}
2012-01-14 07:44:18 +00:00
wcstring_list_t function_get_names ( int get_hidden )
2005-09-20 13:26:39 +00:00
{
2012-01-14 07:44:18 +00:00
std : : set < wcstring > names ;
2012-01-24 04:32:36 +00:00
scoped_lock lock ( functions_lock ) ;
2012-01-14 07:44:18 +00:00
autoload_names ( names , get_hidden ) ;
function_map_t : : const_iterator iter ;
for ( iter = loaded_functions . begin ( ) ; iter ! = loaded_functions . end ( ) ; iter + + ) {
const wcstring & name = iter - > first ;
/* Maybe skip hidden */
if ( ! get_hidden ) {
if ( name . size ( ) = = 0 | | name . at ( 0 ) = = L ' _ ' ) continue ;
}
names . insert ( name ) ;
}
return wcstring_list_t ( names . begin ( ) , names . end ( ) ) ;
2005-09-20 13:26:39 +00:00
}
2005-10-05 22:37:08 +00:00
2007-04-16 20:06:11 +00:00
const wchar_t * function_get_definition_file ( const wchar_t * name )
2006-01-26 14:48:10 +00:00
{
2012-01-24 04:43:39 +00:00
shared_ptr < function_info_t > func = function_get ( name ) ;
return func ? func - > definition_file : NULL ;
2006-01-26 14:48:10 +00:00
}
2007-04-16 20:06:11 +00:00
int function_get_definition_offset ( const wchar_t * name )
2006-01-26 14:48:10 +00:00
{
2012-01-24 04:43:39 +00:00
shared_ptr < function_info_t > func = function_get ( name ) ;
return func ? func - > definition_offset : - 1 ;
2006-01-26 14:48:10 +00:00
}