Some initial work towards resolving nasty fork/pthread issues, and to having a per-parser job list

This commit is contained in:
ridiculousfish 2012-02-27 18:43:24 -08:00
parent 50ee5d28cd
commit fdfa5c0602
13 changed files with 258 additions and 137 deletions

View file

@ -1133,7 +1133,7 @@ static void functions_def( const wcstring &name, wcstring &out )
case EVENT_JOB_ID: case EVENT_JOB_ID:
{ {
job_t *j = job_get( next->param1.job_id ); const job_t *j = job_get( next->param1.job_id );
if( j ) if( j )
append_format( out, L" --on-job-exit %d", j->pgid ); append_format( out, L" --on-job-exit %d", j->pgid );
break; break;

View file

@ -71,7 +71,7 @@ static int cpu_use( job_t *j )
/** /**
Print information about the specified job Print information about the specified job
*/ */
static void builtin_jobs_print( job_t *j, int mode, int header ) static void builtin_jobs_print( const job_t *j, int mode, int header )
{ {
process_t *p; process_t *p;
switch( mode ) switch( mode )
@ -164,7 +164,7 @@ static int builtin_jobs( parser_t &parser, wchar_t **argv )
int found=0; int found=0;
int mode=JOBS_DEFAULT; int mode=JOBS_DEFAULT;
int print_last = 0; int print_last = 0;
job_t *j; const job_t *j;
argc = builtin_count_args( argv ); argc = builtin_count_args( argv );
woptind=0; woptind=0;
@ -270,7 +270,7 @@ static int builtin_jobs( parser_t &parser, wchar_t **argv )
Ignore unconstructed jobs, i.e. ourself. Ignore unconstructed jobs, i.e. ourself.
*/ */
job_iterator_t jobs; job_iterator_t jobs;
job_t *j; const job_t *j;
while ((j = jobs.next())) while ((j = jobs.next()))
{ {
@ -324,7 +324,7 @@ static int builtin_jobs( parser_t &parser, wchar_t **argv )
else else
{ {
job_iterator_t jobs; job_iterator_t jobs;
job_t *j; const job_t *j;
while ((j = jobs.next())) while ((j = jobs.next()))
{ {
/* /*

View file

@ -2058,6 +2058,22 @@ void set_main_thread() {
main_thread_id = pthread_self(); main_thread_id = pthread_self();
} }
/* Notice when we've forked */
static bool is_child_of_fork = false;
static void child_note_forked(void) {
is_child_of_fork = true;
}
bool is_forked_child(void) {
return is_child_of_fork;
}
void setup_fork_guards(void) {
/* Notice when we fork */
pthread_atfork(NULL /* prepare */, NULL /* parent */, child_note_forked);
}
static bool is_main_thread() { static bool is_main_thread() {
assert (main_thread_id != 0); assert (main_thread_id != 0);
return main_thread_id == pthread_self(); return main_thread_id == pthread_self();
@ -2068,7 +2084,14 @@ void assert_is_main_thread(const char *who)
if (! is_main_thread()) { if (! is_main_thread()) {
fprintf(stderr, "Warning: %s called off of main thread. Break on debug_thread_error to debug.\n", who); fprintf(stderr, "Warning: %s called off of main thread. Break on debug_thread_error to debug.\n", who);
debug_thread_error(); debug_thread_error();
sleep(1000); }
}
void assert_is_not_forked_child(const char *who)
{
if (is_forked_child()) {
fprintf(stderr, "Warning: %s called in a forked child. Break on debug_thread_error to debug.\n", who);
debug_thread_error();
} }
} }
@ -2089,3 +2112,25 @@ void assert_is_locked(void *vmutex, const char *who)
pthread_mutex_unlock(mutex); pthread_mutex_unlock(mutex);
} }
} }
void scoped_lock::lock(void) {
assert(! locked);
assert(! is_forked_child());
VOMIT_ON_FAILURE(pthread_mutex_lock(lock_obj));
locked = true;
}
void scoped_lock::unlock(void) {
assert(locked);
assert(! is_forked_child());
VOMIT_ON_FAILURE(pthread_mutex_unlock(lock_obj));
locked = false;
}
scoped_lock::scoped_lock(pthread_mutex_t &mutex) : lock_obj(&mutex), locked(false) {
this->lock();
}
scoped_lock::~scoped_lock() {
if (locked) this->unlock();
}

View file

@ -301,30 +301,16 @@ wcstring to_string(const T &x) {
return stream.str(); return stream.str();
} }
bool is_forked_child();
class scoped_lock { class scoped_lock {
pthread_mutex_t *lock_obj; pthread_mutex_t *lock_obj;
bool locked; bool locked;
public: public:
void lock(void);
void lock(void) { void unlock(void);
assert(! locked); scoped_lock(pthread_mutex_t &mutex);
VOMIT_ON_FAILURE(pthread_mutex_lock(lock_obj)); ~scoped_lock();
locked = true;
}
void unlock(void) {
assert(locked);
VOMIT_ON_FAILURE(pthread_mutex_unlock(lock_obj));
locked = false;
}
scoped_lock(pthread_mutex_t &mutex) : lock_obj(&mutex), locked(false) {
this->lock();
}
~scoped_lock() {
if (locked) this->unlock();
}
}; };
class wcstokenizer { class wcstokenizer {
@ -597,14 +583,13 @@ double timef();
*/ */
void set_main_thread(); void set_main_thread();
/** /** Set up a guard to complain if we try to do certain things (like take a lock) after calling fork */
void setup_fork_guards(void);
*/
/**
*/
/** Return whether we are the child of a fork */
bool is_forked_child(void);
void assert_is_not_forked_child(const char *who);
#define ASSERT_IS_NOT_FORKED_CHILD_TRAMPOLINE(x) assert_is_not_forked_child(x)
#define ASSERT_IS_NOT_FORKED_CHILD() ASSERT_IS_NOT_FORKED_CHILD_TRAMPOLINE(__FUNCTION__)
#endif #endif

View file

@ -831,6 +831,8 @@ static int set_child_group( job_t *j, process_t *p, int print_errors )
*/ */
static pid_t exec_fork() static pid_t exec_fork()
{ {
ASSERT_IS_MAIN_THREAD();
pid_t pid; pid_t pid;
struct timespec pollint; struct timespec pollint;
int i; int i;
@ -1047,6 +1049,7 @@ void exec( parser_t &parser, job_t *j )
if( keepalive.pid == 0 ) if( keepalive.pid == 0 )
{ {
/* Child */
keepalive.pid = getpid(); keepalive.pid = getpid();
set_child_group( j, &keepalive, 1 ); set_child_group( j, &keepalive, 1 );
pause(); pause();
@ -1054,6 +1057,7 @@ void exec( parser_t &parser, job_t *j )
} }
else else
{ {
/* Parent */
set_child_group( j, &keepalive, 0 ); set_child_group( j, &keepalive, 0 );
} }
} }

View file

@ -334,7 +334,7 @@ static int find_process( const wchar_t *proc,
int sz=0; int sz=0;
int found = 0; int found = 0;
job_t *j; const job_t *j;
if( iswnumeric(proc) || (wcslen(proc)==0) ) if( iswnumeric(proc) || (wcslen(proc)==0) )
{ {

View file

@ -260,7 +260,6 @@ static int fish_parse_opt( int argc, char **argv, const char **cmd_ptr )
return my_optind; return my_optind;
} }
/** /**
Calls a bunch of init functions, parses the init files and then Calls a bunch of init functions, parses the init files and then
parses commands from stdin or files, depending on arguments parses commands from stdin or files, depending on arguments
@ -274,7 +273,8 @@ int main( int argc, char **argv )
int my_optind=0; int my_optind=0;
set_main_thread(); set_main_thread();
setup_fork_guards();
wsetlocale( LC_ALL, L"" ); wsetlocale( LC_ALL, L"" );
is_interactive_session=1; is_interactive_session=1;
program_name=L"fish"; program_name=L"fish";

View file

@ -1,4 +1,6 @@
#include "config.h"
#include "iothread.h" #include "iothread.h"
#include "common.h"
#include <pthread.h> #include <pthread.h>
#include <assert.h> #include <assert.h>
#include <errno.h> #include <errno.h>
@ -139,6 +141,8 @@ static void iothread_spawn_if_needed(void) {
} }
int iothread_perform_base(int (*handler)(void *), void (*completionCallback)(void *, int), void *context) { int iothread_perform_base(int (*handler)(void *), void (*completionCallback)(void *, int), void *context) {
ASSERT_IS_MAIN_THREAD();
ASSERT_IS_NOT_FORKED_CHILD();
iothread_init(); iothread_init();
/* Create and initialize a request. */ /* Create and initialize a request. */

View file

@ -371,6 +371,7 @@ parser_t::parser_t(enum parser_type_t type) :
parser_t &parser_t::principal_parser(void) parser_t &parser_t::principal_parser(void)
{ {
ASSERT_IS_NOT_FORKED_CHILD();
ASSERT_IS_MAIN_THREAD(); ASSERT_IS_MAIN_THREAD();
static parser_t parser(PARSER_TYPE_GENERAL); static parser_t parser(PARSER_TYPE_GENERAL);
return parser; return parser;
@ -764,6 +765,12 @@ int parser_t::eval_args( const wchar_t *line, std::vector<completion_t> &args )
tokenizer tok; tokenizer tok;
const bool show_errors = (this->parser_type == PARSER_TYPE_GENERAL || this->parser_type == PARSER_TYPE_ERRORS_ONLY); const bool show_errors = (this->parser_type == PARSER_TYPE_GENERAL || this->parser_type == PARSER_TYPE_ERRORS_ONLY);
expand_flags_t eflags = 0;
if (! show_errors)
eflags |= EXPAND_NO_DESCRIPTIONS;
if (this->parser_type != PARSER_TYPE_GENERAL)
eflags |= EXPAND_SKIP_CMDSUBST;
/* /*
eval_args may be called while evaulating another command, so we eval_args may be called while evaulating another command, so we
save the previous tokenizer and restore it on exit save the previous tokenizer and restore it on exit
@ -799,7 +806,7 @@ int parser_t::eval_args( const wchar_t *line, std::vector<completion_t> &args )
DIE_MEM(); DIE_MEM();
} }
if( expand_string( tmp, args, (show_errors ? 0 : EXPAND_NO_DESCRIPTIONS) ) == EXPAND_ERROR ) if( expand_string( tmp, args, eflags ) == EXPAND_ERROR )
{ {
err_pos=tok_get_pos( &tok ); err_pos=tok_get_pos( &tok );
do_loop=0; do_loop=0;
@ -1210,6 +1217,62 @@ int parser_t::is_help( const wchar_t *s, int min_match ) const
( len >= min_match && (wcsncmp( L"--help", s, len ) == 0) ); ( len >= min_match && (wcsncmp( L"--help", s, len ) == 0) );
} }
job_t *parser_t::job_create(void)
{
job_t *res = new job_t(acquire_job_id());
this->my_job_list.push_front(res);
job_set_flag( res,
JOB_CONTROL,
(job_control_mode==JOB_CONTROL_ALL) ||
((job_control_mode == JOB_CONTROL_INTERACTIVE) && (get_is_interactive())) );
return res;
}
bool parser_t::job_remove( job_t *j )
{
job_list_t::iterator iter = std::find(my_job_list.begin(), my_job_list.end(), j);
if (iter != my_job_list.end()) {
my_job_list.erase(iter);
return true;
} else {
debug( 1, _( L"Job inconsistency" ) );
sanity_lose();
return false;
}
}
void parser_t::job_promote(job_t *job)
{
job_list_t::iterator loc = find(my_job_list.begin(), my_job_list.end(), job);
assert(loc != my_job_list.end());
/* Move the job to the beginning */
my_job_list.splice(my_job_list.begin(), my_job_list, loc);
}
job_t *parser_t::job_get(job_id_t id)
{
job_iterator_t jobs(my_job_list);
job_t *job;
while ((job = jobs.next())) {
if( id <= 0 || job->job_id == id)
return job;
}
return NULL;
}
job_t *parser_t::job_get_from_pid( int pid )
{
job_iterator_t jobs;
job_t *job;
while ((job = jobs.next())) {
if( job->pgid == pid )
return job;
}
return 0;
}
/** /**
Parse options for the specified job Parse options for the specified job
@ -2212,7 +2275,7 @@ void parser_t::eval_job( tokenizer *tok )
{ {
case TOK_STRING: case TOK_STRING:
{ {
j = job_create(); j = this->job_create();
job_set_flag( j, JOB_FOREGROUND, 1 ); job_set_flag( j, JOB_FOREGROUND, 1 );
job_set_flag( j, JOB_TERMINAL, job_get_flag( j, JOB_CONTROL ) ); job_set_flag( j, JOB_TERMINAL, job_get_flag( j, JOB_CONTROL ) );
job_set_flag( j, JOB_TERMINAL, job_get_flag( j, JOB_CONTROL ) \ job_set_flag( j, JOB_TERMINAL, job_get_flag( j, JOB_CONTROL ) \

View file

@ -9,7 +9,6 @@
#include "proc.h" #include "proc.h"
#include "util.h" #include "util.h"
#include "parser.h"
#include "event.h" #include "event.h"
#include "function.h" #include "function.h"
#include <vector> #include <vector>
@ -295,6 +294,9 @@ class parser_t {
/** String index where the current job started. */ /** String index where the current job started. */
int job_start_pos; int job_start_pos;
/** The jobs associated with this parser */
job_list_t my_job_list;
/** /**
Keeps track of how many recursive eval calls have been made. Eval Keeps track of how many recursive eval calls have been made. Eval
doesn't call itself directly, recursion happens on blocks and on doesn't call itself directly, recursion happens on blocks and on
@ -381,46 +383,47 @@ class parser_t {
*/ */
const wchar_t *current_line(); const wchar_t *current_line();
/** /** Returns the current line number */
Returns the current line number
*/
int get_lineno() const; int get_lineno() const;
/** /** Returns the current position in the latest string of the tokenizer. */
Returns the current position in the latest string of the tokenizer.
*/
int get_pos() const; int get_pos() const;
/** /** Returns the position where the current job started in the latest string of the tokenizer. */
Returns the position where the current job started in the latest string of the tokenizer.
*/
int get_job_pos() const; int get_job_pos() const;
/** /** Set the current position in the latest string of the tokenizer. */
Set the current position in the latest string of the tokenizer.
*/
void set_pos( int p); void set_pos( int p);
/** /** Get the string currently parsed */
Get the string currently parsed
*/
const wchar_t *get_buffer() const; const wchar_t *get_buffer() const;
/** Get the list of jobs */
job_list_t &job_list() { return my_job_list; }
/** /** Create block of specified type */
Create block of specified type
*/
void push_block( int type); void push_block( int type);
/** /** Remove the outermost block namespace */
Remove the outermost block namespace
*/
void pop_block(); void pop_block();
/** /** Return a description of the given blocktype */
Return a description of the given blocktype
*/
const wchar_t *get_block_desc( int block ) const; const wchar_t *get_block_desc( int block ) const;
/** Create a job */
job_t *job_create();
/** Removes a job */
bool job_remove(job_t *job);
/** Promotes a job to the front of the list */
void job_promote(job_t *job);
/** Return the job with the specified job id. If id is 0 or less, return the last job used. */
job_t *job_get(int job_id);
/** Returns the job with the given pid */
job_t *job_get_from_pid( int pid );
/** /**
Test if the specified string can be parsed, or if more bytes need Test if the specified string can be parsed, or if more bytes need

132
proc.cpp
View file

@ -95,12 +95,28 @@ static int last_status=0;
*/ */
static sig_atomic_t got_signal=0; static sig_atomic_t got_signal=0;
static std::list<job_t *> s_job_list; bool job_list_is_empty(void)
{
job_list_t &job_list(void) { ASSERT_IS_MAIN_THREAD();
return s_job_list; return parser_t::principal_parser().job_list().empty();
} }
void job_iterator_t::reset()
{
this->current = job_list->begin();
this->end = job_list->end();
}
job_iterator_t::job_iterator_t(job_list_t &jobs) : job_list(&jobs) {
this->reset();
}
job_iterator_t::job_iterator_t() : job_list(&parser_t::principal_parser().job_list()) {
this->reset();
}
int is_interactive_session=0; int is_interactive_session=0;
int is_subshell=0; int is_subshell=0;
int is_block=0; int is_block=0;
@ -140,26 +156,14 @@ void proc_init()
*/ */
static int job_remove( job_t *j ) static int job_remove( job_t *j )
{ {
job_list_t &jobs = job_list(); ASSERT_IS_MAIN_THREAD();
job_list_t::iterator iter = find(jobs.begin(), jobs.end(), j); return parser_t::principal_parser().job_remove(j);
if (iter != jobs.end()) {
jobs.erase(iter);
return 1;
} else {
debug( 1, _( L"Job inconsistency" ) );
sanity_lose();
return 0;
}
} }
void job_promote(job_t *job) void job_promote(job_t *job)
{ {
job_list_t &jobs = job_list(); ASSERT_IS_MAIN_THREAD();
job_list_t::iterator loc = find(jobs.begin(), jobs.end(), job); parser_t::principal_parser().job_promote(job);
assert(loc != jobs.end());
/* Move the job to the beginning */
jobs.splice(jobs.begin(), jobs, loc);
} }
@ -201,13 +205,13 @@ void process_t::set_argv(const wcstring_list_t &argv) {
void proc_destroy() void proc_destroy()
{ {
event.arguments.reset(NULL); event.arguments.reset(NULL);
job_list_t &jobs = job_list(); job_list_t &jobs = parser_t::principal_parser().job_list();
while( ! jobs.empty() ) while( ! jobs.empty() )
{ {
job_t *job = jobs.front(); job_t *job = jobs.front();
debug( 2, L"freeing leaked job %ls", job->command_cstr() ); debug( 2, L"freeing leaked job %ls", job->command_cstr() );
job_free( job ); job_free( job );
} }
} }
void proc_set_last_status( int s ) void proc_set_last_status( int s )
@ -220,47 +224,59 @@ int proc_get_last_status()
return last_status; return last_status;
} }
job_t *job_create() /* Basic thread safe job IDs. The vector consumed_job_ids has a true value wherever the job ID corresponding to that slot is in use. The job ID corresponding to slot 0 is 1. */
{ static pthread_mutex_t job_id_lock = PTHREAD_MUTEX_INITIALIZER;
int free_id=1; static std::vector<bool> consumed_job_ids;
while( job_get( free_id ) != 0 )
free_id++;
job_t *res = new job_t(free_id);
job_list().push_front(res);
job_set_flag( res,
JOB_CONTROL,
(job_control_mode==JOB_CONTROL_ALL) ||
((job_control_mode == JOB_CONTROL_INTERACTIVE) && (is_interactive)) );
// if( res->job_id > 2 ) job_id_t acquire_job_id(void)
// fwprintf( stderr, L"Create job %d\n", res->job_id ); {
return res; scoped_lock lock(job_id_lock);
/* Find the index of the first 0 slot */
std::vector<bool>::iterator slot = std::find(consumed_job_ids.begin(), consumed_job_ids.end(), false);
if (slot != consumed_job_ids.end())
{
/* We found a slot. Note that slot 0 corresponds to job ID 1. */
*slot = true;
return slot - consumed_job_ids.begin() + 1;
}
else
{
/* We did not find a slot; create a new slot. The size of the vector is now the job ID (since it is one larger than the slot). */
consumed_job_ids.push_back(true);
return (job_id_t)consumed_job_ids.size();
}
} }
void release_job_id(job_id_t jid)
job_t *job_get( int id )
{ {
job_iterator_t jobs; assert(jid > 0);
job_t *job; scoped_lock lock(job_id_lock);
while ((job = jobs.next())) { size_t slot = (size_t)(jid - 1), count = consumed_job_ids.size();
if( id <= 0 || job->job_id == id)
return job; /* Make sure this slot is within our vector and is currently set to consumed */
} assert(slot < count);
return NULL; assert(consumed_job_ids.at(slot) == true);
/* Clear it and then resize the vector to eliminate unused trailing job IDs */
consumed_job_ids.at(slot) = false;
while (count--) {
if (consumed_job_ids.at(count))
break;
}
consumed_job_ids.resize(count + 1);
}
job_t *job_get( job_id_t id )
{
ASSERT_IS_MAIN_THREAD();
return parser_t::principal_parser().job_get(id);
} }
job_t *job_get_from_pid( int pid ) job_t *job_get_from_pid( int pid )
{ {
job_iterator_t jobs; ASSERT_IS_MAIN_THREAD();
job_t *job; return parser_t::principal_parser().job_get_from_pid(pid);
while ((job = jobs.next())) {
if( job->pgid == pid )
return job;
}
return 0;
} }
@ -308,7 +324,7 @@ void job_set_flag( job_t *j, int flag, int set )
j->flags = j->flags & (0xffffffff ^ flag); j->flags = j->flags & (0xffffffff ^ flag);
} }
int job_get_flag( job_t *j, int flag ) int job_get_flag( const job_t *j, int flag )
{ {
return j->flags&flag?1:0; return j->flags&flag?1:0;
} }
@ -352,7 +368,7 @@ int job_signal( job_t *j, int signal )
Store the status of the process pid that was returned by waitpid. Store the status of the process pid that was returned by waitpid.
Return 0 if all went well, nonzero otherwise. Return 0 if all went well, nonzero otherwise.
*/ */
static void mark_process_status( job_t *j, static void mark_process_status( const job_t *j,
process_t *p, process_t *p,
int status ) int status )
{ {
@ -399,7 +415,7 @@ static void mark_process_status( job_t *j,
static void handle_child_status( pid_t pid, int status ) static void handle_child_status( pid_t pid, int status )
{ {
int found_proc = 0; int found_proc = 0;
job_t *j=0; const job_t *j=0;
process_t *p=0; process_t *p=0;
// char mess[MESS_SIZE]; // char mess[MESS_SIZE];
found_proc = 0; found_proc = 0;

29
proc.h
View file

@ -279,11 +279,15 @@ class process_t
A struct represeting a job. A job is basically a pipeline of one A struct represeting a job. A job is basically a pipeline of one
or more processes and a couple of flags. or more processes and a couple of flags.
*/ */
typedef int job_id_t;
job_id_t acquire_job_id(void);
void release_job_id(job_id_t jobid);
class job_t class job_t
{ {
public: public:
job_t(int jobid) : job_t(job_id_t jobid) :
command(), command(),
first_process(NULL), first_process(NULL),
pgid(0), pgid(0),
@ -303,6 +307,7 @@ class job_t
delete data; delete data;
data = tmp; data = tmp;
} }
release_job_id(job_id);
} }
@ -339,7 +344,7 @@ class job_t
unique identifier of the job within this shell, and is unique identifier of the job within this shell, and is
used e.g. in process expansion. used e.g. in process expansion.
*/ */
const int job_id; const job_id_t job_id;
/** /**
List of all IO redirections for this job. This linked list is allocated via new, and owned by the object, which should delete them. List of all IO redirections for this job. This linked list is allocated via new, and owned by the object, which should delete them.
@ -384,20 +389,17 @@ extern int is_login;
extern int is_event; extern int is_event;
/** List of all living jobs */
typedef std::list<job_t *> job_list_t; typedef std::list<job_t *> job_list_t;
job_list_t &job_list(void);
bool job_list_is_empty(void);
/** A class to aid iteration over jobs list */ /** A class to aid iteration over jobs list */
class job_iterator_t { class job_iterator_t {
job_list_t * const job_list;
job_list_t::iterator current, end; job_list_t::iterator current, end;
public: public:
void reset(void) { void reset(void);
job_list_t &jobs = job_list();
this->current = jobs.begin();
this->end = jobs.end();
}
job_t *next() { job_t *next() {
job_t *job = NULL; job_t *job = NULL;
@ -408,9 +410,8 @@ class job_iterator_t {
return job; return job;
} }
job_iterator_t() { job_iterator_t(job_list_t &jobs);
this->reset(); job_iterator_t();
}
}; };
/** /**
@ -452,7 +453,7 @@ void job_set_flag( job_t *j, int flag, int set );
/** /**
Returns one if the specified flag is set in the specified job, 0 otherwise. Returns one if the specified flag is set in the specified job, 0 otherwise.
*/ */
int job_get_flag( job_t *j, int flag ); int job_get_flag( const job_t *j, int flag );
/** /**
Sets the status of the last process to exit Sets the status of the last process to exit
@ -483,7 +484,7 @@ job_t *job_create();
Return the job with the specified job id. Return the job with the specified job id.
If id is 0 or less, return the last job used. If id is 0 or less, return the last job used.
*/ */
job_t *job_get(int id); job_t *job_get(job_id_t id);
/** /**

View file

@ -2481,7 +2481,7 @@ static void reader_super_highlight_me_plenty( int match_highlight_pos )
int exit_status() int exit_status()
{ {
if( get_is_interactive() ) if( get_is_interactive() )
return job_list().empty() && data->end_loop; return job_list_is_empty() && data->end_loop;
else else
return end_loop; return end_loop;
} }