ridiculousfish 2012-05-11 18:59:38 -07:00
parent 2d3d6e1c17
commit c15975113a
5 changed files with 79 additions and 43 deletions

View file

@ -162,6 +162,7 @@ static unsigned int kCompleteOrder = 0;
typedef std::list<complete_entry_opt_t> option_list_t; typedef std::list<complete_entry_opt_t> option_list_t;
class completion_entry_t class completion_entry_t
{ {
public:
/** List of all options */ /** List of all options */
option_list_t options; option_list_t options;
@ -183,9 +184,12 @@ class completion_entry_t
const unsigned int order; const unsigned int order;
/** Getters for option list. */ /** Getters for option list. */
option_list_t &get_options();
const option_list_t &get_options() const; const option_list_t &get_options() const;
/** Adds or removes an option. */
void add_option(const complete_entry_opt_t &opt);
bool remove_option(wchar_t short_opt, const wchar_t *long_opt);
/** Getter for short_opt_str. */ /** Getter for short_opt_str. */
wcstring &get_short_opt_str(); wcstring &get_short_opt_str();
const wcstring &get_short_opt_str() const; const wcstring &get_short_opt_str() const;
@ -226,9 +230,10 @@ static pthread_mutex_t completion_lock = PTHREAD_MUTEX_INITIALIZER;
/** The lock that guards the options list of individual completion entries. If both completion_lock and completion_entry_lock are to be taken, completion_lock must be taken first. */ /** The lock that guards the options list of individual completion entries. If both completion_lock and completion_entry_lock are to be taken, completion_lock must be taken first. */
static pthread_mutex_t completion_entry_lock = PTHREAD_MUTEX_INITIALIZER; static pthread_mutex_t completion_entry_lock = PTHREAD_MUTEX_INITIALIZER;
option_list_t &completion_entry_t::get_options() {
void completion_entry_t::add_option(const complete_entry_opt_t &opt) {
ASSERT_IS_LOCKED(completion_entry_lock); ASSERT_IS_LOCKED(completion_entry_lock);
return options; options.push_front(opt);
} }
const option_list_t &completion_entry_t::get_options() const { const option_list_t &completion_entry_t::get_options() const {
@ -434,16 +439,15 @@ void complete_add( const wchar_t *cmd,
/* Lock the lock that allows us to edit the completion entry list */ /* Lock the lock that allows us to edit the completion entry list */
scoped_lock lock(completion_lock); scoped_lock lock(completion_lock);
completion_entry_t *c;
c = complete_get_exact_entry( cmd, cmd_is_path );
/* Lock the lock that allows us to edit individual completion entries */ /* Lock the lock that allows us to edit individual completion entries */
scoped_lock lock2(completion_entry_lock); scoped_lock lock2(completion_entry_lock);
option_list_t &options = c->get_options(); completion_entry_t *c;
options.push_front(complete_entry_opt_t()); c = complete_get_exact_entry( cmd, cmd_is_path );
complete_entry_opt_t &opt = options.front();
/* Create our new option */
complete_entry_opt_t opt;
if( short_opt != L'\0' ) if( short_opt != L'\0' )
{ {
int len = 1 + ((result_mode & NO_COMMON) != 0); int len = 1 + ((result_mode & NO_COMMON) != 0);
@ -464,6 +468,8 @@ void complete_add( const wchar_t *cmd,
if (long_opt) opt.long_opt = long_opt; if (long_opt) opt.long_opt = long_opt;
if (desc) opt.desc = desc; if (desc) opt.desc = desc;
opt.flags = flags; opt.flags = flags;
c->add_option(opt);
} }
/** /**
@ -471,21 +477,17 @@ void complete_add( const wchar_t *cmd,
specified short / long option strings. Returns true if it is now specified short / long option strings. Returns true if it is now
empty and should be deleted, false if it's not empty. Must be called while locked. empty and should be deleted, false if it's not empty. Must be called while locked.
*/ */
static bool complete_remove_entry( completion_entry_t *e, wchar_t short_opt, const wchar_t *long_opt ) bool completion_entry_t::remove_option( wchar_t short_opt, const wchar_t *long_opt )
{ {
ASSERT_IS_LOCKED(completion_lock); ASSERT_IS_LOCKED(completion_lock);
ASSERT_IS_LOCKED(completion_entry_lock); ASSERT_IS_LOCKED(completion_entry_lock);
option_list_t &options = e->get_options();
if(( short_opt == 0 ) && (long_opt == 0 ) ) if(( short_opt == 0 ) && (long_opt == 0 ) )
{ {
options.clear(); this->options.clear();
} }
else else
{ {
/* Lock the lock that allows us to edit individual completion entries */ for (option_list_t::iterator iter = this->options.begin(); iter != this->options.end(); )
scoped_lock lock2(completion_entry_lock);
for (option_list_t::iterator iter = options.begin(); iter != options.end(); )
{ {
complete_entry_opt_t &o = *iter; complete_entry_opt_t &o = *iter;
if(short_opt==o.short_opt || long_opt == o.long_opt) if(short_opt==o.short_opt || long_opt == o.long_opt)
@ -497,7 +499,7 @@ static bool complete_remove_entry( completion_entry_t *e, wchar_t short_opt, con
*/ */
if( o.short_opt ) if( o.short_opt )
{ {
wcstring &short_opt_str = e->get_short_opt_str(); wcstring &short_opt_str = this->get_short_opt_str();
size_t idx = short_opt_str.find(o.short_opt); size_t idx = short_opt_str.find(o.short_opt);
if (idx != wcstring::npos) if (idx != wcstring::npos)
{ {
@ -510,7 +512,7 @@ static bool complete_remove_entry( completion_entry_t *e, wchar_t short_opt, con
} }
/* Destroy this option and go to the next one */ /* Destroy this option and go to the next one */
iter = options.erase(iter); iter = this->options.erase(iter);
} }
else else
{ {
@ -519,7 +521,7 @@ static bool complete_remove_entry( completion_entry_t *e, wchar_t short_opt, con
} }
} }
} }
return options.empty(); return this->options.empty();
} }
@ -536,7 +538,7 @@ void complete_remove( const wchar_t *cmd,
completion_entry_set_t::iterator iter = completion_set.find(&tmp_entry); completion_entry_set_t::iterator iter = completion_set.find(&tmp_entry);
if (iter != completion_set.end()) { if (iter != completion_set.end()) {
completion_entry_t *entry = *iter; completion_entry_t *entry = *iter;
bool delete_it = complete_remove_entry(entry, short_opt, long_opt); bool delete_it = entry->remove_option(short_opt, long_opt);
if (delete_it) { if (delete_it) {
/* Delete this entry */ /* Delete this entry */
completion_set.erase(iter); completion_set.erase(iter);
@ -1244,12 +1246,16 @@ void complete_load( const wcstring &name, bool reload )
previous option popt. Insert results into comp_out. Return 0 if file previous option popt. Insert results into comp_out. Return 0 if file
completion should be disabled, 1 otherwise. completion should be disabled, 1 otherwise.
*/ */
struct local_options_t {
wcstring short_opt_str;
option_list_t options;
};
bool completer_t::complete_param( const wcstring &scmd_orig, const wcstring &spopt, const wcstring &sstr, bool use_switches) bool completer_t::complete_param( const wcstring &scmd_orig, const wcstring &spopt, const wcstring &sstr, bool use_switches)
{ {
const wchar_t * const cmd_orig = scmd_orig.c_str(), * const popt = spopt.c_str(), * const str = sstr.c_str(); const wchar_t * const cmd_orig = scmd_orig.c_str(), * const popt = spopt.c_str(), * const str = sstr.c_str();
int use_common=1, use_files=1; bool use_common=1, use_files=1;
wcstring cmd, path; wcstring cmd, path;
parse_cmd_string(cmd_orig, path, cmd); parse_cmd_string(cmd_orig, path, cmd);
@ -1267,19 +1273,33 @@ bool completer_t::complete_param( const wcstring &scmd_orig, const wcstring &spo
} }
} }
/* Make a list of lists of all options that we care about */
std::vector<local_options_t> all_options;
{
scoped_lock lock(completion_lock); scoped_lock lock(completion_lock);
scoped_lock lock2(completion_entry_lock); scoped_lock lock2(completion_entry_lock);
for (completion_entry_set_t::const_iterator iter = completion_set.begin(); iter != completion_set.end(); ++iter) for (completion_entry_set_t::const_iterator iter = completion_set.begin(); iter != completion_set.end(); ++iter)
{ {
const completion_entry_t *i = *iter; const completion_entry_t *i = *iter;
const wcstring &match = i->cmd_is_path ? path : cmd; const wcstring &match = i->cmd_is_path ? path : cmd;
if (! wildcard_match(match, i->cmd))
if( ( (!wildcard_match( match, i->cmd ) ) ) )
{ {
continue; continue;
} }
const option_list_t &options = i->get_options(); /* Copy all of their options into our list */
all_options.push_back(local_options_t());
all_options.back().short_opt_str = i->get_short_opt_str();
all_options.back().options = i->get_options(); //Oof, this is a lot of copying
}
}
/* Now release the lock and test each option that we captured above.
We have to do this outside the lock because callouts (like the condition) may add or remove completions.
See https://github.com/ridiculousfish/fishfish/issues/2 */
for (std::vector<local_options_t>::const_iterator iter = all_options.begin(); iter != all_options.end(); iter++)
{
const option_list_t &options = iter->options;
use_common=1; use_common=1;
if( use_switches ) if( use_switches )
{ {
@ -1294,8 +1314,8 @@ bool completer_t::complete_param( const wcstring &scmd_orig, const wcstring &spo
wchar_t *arg; wchar_t *arg;
if( (arg=param_match2( o, str ))!=0 && this->condition_test( o->condition )) if( (arg=param_match2( o, str ))!=0 && this->condition_test( o->condition ))
{ {
use_common &= ((o->result_mode & NO_COMMON )==0); if (o->result_mode & NO_COMMON) use_common = false;
use_files &= ((o->result_mode & NO_FILES )==0); if (o->result_mode & NO_FILES) use_files = false;
complete_from_args( arg, o->comp, o->localized_desc(), o->flags ); complete_from_args( arg, o->comp, o->localized_desc(), o->flags );
} }
@ -1318,8 +1338,8 @@ bool completer_t::complete_param( const wcstring &scmd_orig, const wcstring &spo
if( param_match_old( o, popt ) && this->condition_test( o->condition )) if( param_match_old( o, popt ) && this->condition_test( o->condition ))
{ {
old_style_match = 1; old_style_match = 1;
use_common &= ((o->result_mode & NO_COMMON )==0); if (o->result_mode & NO_COMMON) use_common = false;
use_files &= ((o->result_mode & NO_FILES )==0); if (o->result_mode & NO_FILES) use_files = false;
complete_from_args( str, o->comp, o->localized_desc(), o->flags ); complete_from_args( str, o->comp, o->localized_desc(), o->flags );
} }
} }
@ -1345,8 +1365,8 @@ bool completer_t::complete_param( const wcstring &scmd_orig, const wcstring &spo
if( param_match( o, popt ) && this->condition_test( o->condition )) if( param_match( o, popt ) && this->condition_test( o->condition ))
{ {
use_common &= ((o->result_mode & NO_COMMON )==0); if (o->result_mode & NO_COMMON) use_common = false;
use_files &= ((o->result_mode & NO_FILES )==0); if (o->result_mode & NO_FILES) use_files = false;
complete_from_args( str, o->comp.c_str(), o->localized_desc(), o->flags ); complete_from_args( str, o->comp.c_str(), o->localized_desc(), o->flags );
} }
@ -1382,7 +1402,7 @@ bool completer_t::complete_param( const wcstring &scmd_orig, const wcstring &spo
Check if the short style option matches Check if the short style option matches
*/ */
if( o->short_opt != L'\0' && if( o->short_opt != L'\0' &&
short_ok( str, o->short_opt, i->get_short_opt_str() ) ) short_ok(str, o->short_opt, iter->short_opt_str))
{ {
const wchar_t *desc = o->localized_desc(); const wchar_t *desc = o->localized_desc();
wchar_t completion[2]; wchar_t completion[2];

0
tests/test6.err Normal file
View file

11
tests/test6.in Executable file
View file

@ -0,0 +1,11 @@
# Test that conditions that add or remove completions don't deadlock, etc.
# We actually encountered some case that was effectively like this (Issue 2 in github)
complete --command AAAA -l abcd --condition 'complete -c AAAA -l efgh'
complete -C'AAAA -'
complete -C'AAAA -'
complete --command BBBB -l abcd --condition 'complete -e --command BBBB -l abcd'
complete -C'BBBB -'
complete -C'BBBB -'

4
tests/test6.out Normal file
View file

@ -0,0 +1,4 @@
--abcd
--efgh
--abcd
--abcd

1
tests/test6.status Normal file
View file

@ -0,0 +1 @@
0