2016-04-19 04:35:53 +00:00
|
|
|
// The classes responsible for autoloading functions and completions.
|
2012-01-25 19:47:45 +00:00
|
|
|
#ifndef FISH_AUTOLOAD_H
|
|
|
|
#define FISH_AUTOLOAD_H
|
2012-01-25 08:36:55 +00:00
|
|
|
|
2015-07-25 15:14:25 +00:00
|
|
|
#include <pthread.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <time.h>
|
2012-01-25 08:36:55 +00:00
|
|
|
#include <set>
|
2016-04-21 06:00:54 +00:00
|
|
|
|
2012-01-25 08:36:55 +00:00
|
|
|
#include "common.h"
|
2012-02-06 04:54:41 +00:00
|
|
|
#include "lru.h"
|
2012-01-25 08:36:55 +00:00
|
|
|
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Record of an attempt to access a file.
|
2016-04-19 04:35:53 +00:00
|
|
|
struct file_access_attempt_t {
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Modification time of the file
|
2016-06-06 02:24:23 +00:00
|
|
|
time_t mod_time;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// When we last checked the file
|
2016-06-06 02:24:23 +00:00
|
|
|
time_t last_checked;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Whether or not we believe we can access this file
|
2016-06-06 02:24:23 +00:00
|
|
|
bool accessible;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// The access attempt is stale
|
2016-06-06 02:24:23 +00:00
|
|
|
bool stale;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// If we cannot access the file, the error code encountered.
|
2016-06-06 02:24:23 +00:00
|
|
|
int error;
|
2012-01-25 08:36:55 +00:00
|
|
|
};
|
2012-01-25 19:47:45 +00:00
|
|
|
file_access_attempt_t access_file(const wcstring &path, int mode);
|
2012-01-25 08:36:55 +00:00
|
|
|
|
2017-01-27 20:18:16 +00:00
|
|
|
struct autoload_function_t {
|
|
|
|
explicit autoload_function_t(bool placeholder)
|
2017-01-30 01:56:03 +00:00
|
|
|
: access(), is_loaded(false), is_placeholder(placeholder), is_internalized(false) {}
|
2016-06-06 04:30:24 +00:00
|
|
|
|
|
|
|
/// The last access attempt recorded
|
2016-06-06 02:24:23 +00:00
|
|
|
file_access_attempt_t access;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Have we actually loaded this function?
|
2016-06-06 02:24:23 +00:00
|
|
|
bool is_loaded;
|
|
|
|
/// Whether we are a placeholder that stands in for "no such function". If this is true, then
|
|
|
|
/// is_loaded must be false.
|
2016-04-19 04:35:53 +00:00
|
|
|
bool is_placeholder;
|
2016-06-06 02:24:23 +00:00
|
|
|
/// Whether this function came from a builtin "internalized" script.
|
2016-04-19 04:35:53 +00:00
|
|
|
bool is_internalized;
|
2012-01-26 02:59:35 +00:00
|
|
|
};
|
|
|
|
|
2016-04-19 04:35:53 +00:00
|
|
|
struct builtin_script_t {
|
2012-12-11 00:23:08 +00:00
|
|
|
const wchar_t *name;
|
|
|
|
const char *def;
|
2012-12-09 05:41:38 +00:00
|
|
|
};
|
2012-01-26 02:59:35 +00:00
|
|
|
|
2012-07-21 03:39:31 +00:00
|
|
|
class env_vars_snapshot_t;
|
2012-01-26 02:59:35 +00:00
|
|
|
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Class representing a path from which we can autoload and the autoloaded contents.
|
2017-01-27 20:18:16 +00:00
|
|
|
class autoload_t : public lru_cache_t<autoload_t, autoload_function_t> {
|
2016-04-19 04:35:53 +00:00
|
|
|
private:
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Lock for thread safety.
|
2012-01-28 22:56:13 +00:00
|
|
|
pthread_mutex_t lock;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// The environment variable name.
|
2012-01-26 02:59:35 +00:00
|
|
|
const wcstring env_var_name;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Builtin script array.
|
2012-01-26 02:59:35 +00:00
|
|
|
const struct builtin_script_t *const builtin_scripts;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Builtin script count.
|
2012-01-26 02:59:35 +00:00
|
|
|
const size_t builtin_script_count;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// The path from which we most recently autoloaded.
|
2012-03-31 22:17:14 +00:00
|
|
|
wcstring last_path;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// the most reecently autoloaded path, tokenized (split on separators).
|
2014-02-24 21:06:26 +00:00
|
|
|
wcstring_list_t last_path_tokenized;
|
2016-06-06 04:30:24 +00:00
|
|
|
/// A table containing all the files that are currently being loaded.
|
|
|
|
/// This is here to help prevent recursion.
|
2012-01-26 02:59:35 +00:00
|
|
|
std::set<wcstring> is_loading_set;
|
2017-01-30 02:56:55 +00:00
|
|
|
// Function invoked when a command is removed
|
|
|
|
typedef void (*command_removed_function_t)(const wcstring &);
|
|
|
|
const command_removed_function_t command_removed;
|
2012-01-26 02:59:35 +00:00
|
|
|
|
2017-01-27 20:18:16 +00:00
|
|
|
void remove_all_functions() { this->evict_all_nodes(); }
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-04-19 04:35:53 +00:00
|
|
|
bool locate_file_and_maybe_load_it(const wcstring &cmd, bool really_load, bool reload,
|
|
|
|
const wcstring_list_t &path_list);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-04-19 04:35:53 +00:00
|
|
|
autoload_function_t *get_autoloaded_function_with_creation(const wcstring &cmd,
|
|
|
|
bool allow_eviction);
|
|
|
|
|
|
|
|
public:
|
2017-01-27 20:18:16 +00:00
|
|
|
// CRTP override
|
|
|
|
void entry_was_evicted(wcstring key, autoload_function_t node);
|
|
|
|
|
|
|
|
// Create an autoload_t for the given environment variable name.
|
2017-01-30 02:56:55 +00:00
|
|
|
autoload_t(const wcstring &env_var_name_var,
|
|
|
|
command_removed_function_t callback,
|
|
|
|
const builtin_script_t *scripts = NULL,
|
|
|
|
size_t script_count = 0);
|
2016-04-19 04:35:53 +00:00
|
|
|
|
2017-01-30 02:56:55 +00:00
|
|
|
~autoload_t();
|
2016-06-06 04:30:24 +00:00
|
|
|
|
|
|
|
/// Autoload the specified file, if it exists in the specified path. Do not load it multiple
|
|
|
|
/// times unless its timestamp changes or parse_util_unload is called.
|
|
|
|
/// Autoloading one file may unload another.
|
|
|
|
/// @param cmd the filename to search for. The suffix '.fish' is always added to this name
|
|
|
|
/// @param reload wheter to recheck file timestamps on already loaded files
|
2012-11-19 00:30:30 +00:00
|
|
|
int load(const wcstring &cmd, bool reload);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Check whether we have tried loading the given command. Does not do any I/O.
|
2012-11-19 00:30:30 +00:00
|
|
|
bool has_tried_loading(const wcstring &cmd);
|
2012-01-26 02:59:35 +00:00
|
|
|
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Tell the autoloader that the specified file, in the specified path, is no longer loaded.
|
|
|
|
/// Returns non-zero if the file was removed, zero if the file had not yet been loaded
|
2012-11-19 00:30:30 +00:00
|
|
|
int unload(const wcstring &cmd);
|
2012-11-18 10:23:22 +00:00
|
|
|
|
2016-06-06 04:30:24 +00:00
|
|
|
/// Check whether the given command could be loaded, but do not load it.
|
2012-11-19 00:30:30 +00:00
|
|
|
bool can_load(const wcstring &cmd, const env_vars_snapshot_t &vars);
|
2012-01-26 02:59:35 +00:00
|
|
|
};
|
2012-01-25 08:36:55 +00:00
|
|
|
#endif
|