Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Introduce 'function' built-in. #77

Open
wants to merge 27 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
faab6cc
Introduce 'function' built-in
Krush206 Jul 7, 2023
b683769
sh.c: Sgoal shouldn't be assigned funcmain
Krush206 Jul 7, 2023
4d90ff3
sh.c: cleanup_push() raises SIGSEGV
Krush206 Jul 8, 2023
715a882
fargv should be freed from dofunction()
Krush206 Jul 8, 2023
5e37bfc
Move function tests to srcunit()
Krush206 Aug 5, 2023
235b358
Better function parsing
Krush206 Nov 21, 2023
4a54e05
Create tests for 'function' built-in
Krush206 Nov 21, 2023
5bc9252
sh.c: call cleanup_push() properly
Krush206 Nov 21, 2023
b1e4135
Documentation for 'function' built-in
Krush206 Nov 21, 2023
ede350d
Rename ERR_DOLFUNC to ERR_FUNC
Krush206 Nov 21, 2023
fc7dbd3
sh.func.c: prefer dolzero over ffile
Krush206 Nov 21, 2023
d762fba
sh.func.c: allocate memory properly
Krush206 Nov 21, 2023
951264d
sh.func.c: assign NULL after deallocating fargv
Krush206 Nov 22, 2023
7d6eef8
Prefer srcfile() over dosource()
Krush206 Dec 3, 2023
c60a4c2
Functions should work for sourced scripts.
Krush206 Feb 14, 2024
954cfd6
Rethought on functions
Krush206 May 18, 2024
15d4e30
Do not fork & fix interrupts
Krush206 Jun 2, 2024
042e84b
Refactor
Krush206 Jun 3, 2024
2fa7967
tcsh.man.in: correction & additional information
Krush206 Jun 3, 2024
b950b85
sh.err.c: recursion/nest
Krush206 Jun 3, 2024
d840e89
Create aliases for functions
Krush206 Jun 3, 2024
22d3150
sh.func.c: replace setexit with cleanup_push
Krush206 Jun 4, 2024
adea1b6
Avoid SIGPIPE on doexit
Krush206 Jun 4, 2024
d0167d4
sh.func.c: don't catch OLDSTD
Krush206 Jun 4, 2024
911ed6e
22d3150: move remaining parts to st_restore
Krush206 Jun 6, 2024
56e5e7d
911ed6e: more remaining parts
Krush206 Jun 7, 2024
c8b8f51
Prefer dozip over doreturn
Krush206 Jun 13, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Functions should work for sourced scripts.
Next, make a table of functions so they can be called globally. Currently, it isn't possible to call functions from sourced files other than the current.
  • Loading branch information
Krush206 committed Jun 1, 2024
commit c60a4c2ba50ebc0ea0073a622d4260569f9ca892
51 changes: 42 additions & 9 deletions sh.c
Original file line number Diff line number Diff line change
Expand Up @@ -111,7 +111,7 @@ int exitset = 0;
static time_t chktim; /* Time mail last checked */
char *progname;
int tcsh;
struct funcargs *fargv = NULL;
struct funccurr fcurr;

/*
* This preserves the input state of the shell. It is used by
Expand Down Expand Up @@ -1725,6 +1725,7 @@ static void
srcunit(int unit, int onlyown, int hflg, Char **av)
{
struct saved_state st;
struct funcargs *fargv;

st.SHIN = -1; /* st_restore checks this */

Expand Down Expand Up @@ -1766,15 +1767,17 @@ srcunit(int unit, int onlyown, int hflg, Char **av)
* then seek for an ending exit on the requested label.
* Function arguments are passed to STRargv.
* STRargv is reset after the function is done. */
if (fargv) {
if (fcurr.ready) {
Char funcexit[] = { 'e', 'x', 'i', 't', 0 },
funcmain[] = { 'm', 'a', 'i', 'n', 0 };
*funcmain = fcurr.ffile ?
strsave(str2short(fcurr.ffile->file)) :
ffile;
struct Strbuf aword = Strbuf_INIT;

Sgoal = fargv->v[0];
Stype = TC_GOTO;
fargv->eof = 0;

if (!fargv->prev)
fcurr.eof = 0;
if (!(fargv = fcurr.fargv)->prev || fcurr.src)
while (1) {
(void) getword(&aword);
Strbuf_terminate(&aword);
Expand All @@ -1799,6 +1802,7 @@ srcunit(int unit, int onlyown, int hflg, Char **av)
}
last = 1;
}
fcurr.src = 0;

break;
}
Expand All @@ -1807,6 +1811,8 @@ srcunit(int unit, int onlyown, int hflg, Char **av)

(void) getword(NULL);
}
if (funcmain != ffile)
xfree(funcmain);

setq(STRargv, &fargv->v[1], &shvhed, VAR_READWRITE);
gotolab(fargv->v[0]);
Expand All @@ -1826,11 +1832,11 @@ srcunit(int unit, int onlyown, int hflg, Char **av)
if (eq(aword.s, funcexit)) {
int last = 1, eof = 0;

fargv->eof = 1;
fcurr.eof = 1;
while (1) {
do {
(void) getword(NULL);
if ((intptr_t) getword(&aword) == (intptr_t) &fargv) {
if (getword(&aword) == (1 << 1)) {
Strbuf_terminate(&aword);
eof = 1;
break;
Expand Down Expand Up @@ -2294,6 +2300,7 @@ dosource(Char **t, struct command *c)
Char *f;
int hflg = 0;
char *file;
struct funcfile **ffile = NULL;

USE(c);
t++;
Expand All @@ -2309,13 +2316,39 @@ dosource(Char **t, struct command *c)
}

f = globone(*t++, G_ERROR);
file = strsave(short2str(f));
fcurr.file = file = strsave(short2str(f));
cleanup_push(file, xfree);
xfree(f);
t = glob_all_or_error(t);
cleanup_push(t, blk_cleanup);
if (fcurr.fargv) {
if (*(ffile = &fcurr.ffile)) {
(*ffile)->next = malloc(sizeof **ffile);
(*ffile)->next->prev = *ffile;
*ffile = (*ffile)->next;
(*ffile)->file = fcurr.file;
} else {
*ffile = malloc(sizeof **ffile);
(*ffile)->prev = NULL;
(*ffile)->file = fcurr.file;
}
}

fcurr.ready = 0;
fcurr.src = 1;
if ((!srcfile(file, 0, hflg, t)) && (!hflg) && (!bequiet))
stderror(ERR_SYSTEM, file, strerror(errno));
if (ffile) {
if ((*ffile)->prev) {
*ffile = (*ffile)->prev;
free((*ffile)->next);
fcurr.file = (*ffile)->file;
} else {
fcurr.file = (*ffile)->file = NULL;
free(*ffile);
*ffile = NULL;
}
}
cleanup_until(file);
}

Expand Down
38 changes: 23 additions & 15 deletions sh.func.c
Original file line number Diff line number Diff line change
Expand Up @@ -1096,8 +1096,8 @@ getword(struct Strbuf *wp)
break;

case TC_EXIT:
if (fargv->eof)
return (intptr_t) &fargv;
if (fcurr.eof)
return 1 << 1;
setname(short2str(Sgoal));
stderror(ERR_NAME | ERR_NOTFOUND, "exit");
break;
Expand Down Expand Up @@ -2729,16 +2729,19 @@ getYN(const char *prompt)
void
dofunction(Char **v, struct command *t)
{
char *file;
struct funcargs **fargv;

if (!dolzero)
stderror(ERR_FUNC);

if (fargv) {
fargv->next = malloc(sizeof *fargv);
fargv->next->prev = fargv;
fargv = fargv->next;
if (*(fargv = &fcurr.fargv)) {
(*fargv)->next = malloc(sizeof **fargv);
(*fargv)->next->prev = *fargv;
*fargv = (*fargv)->next;
} else {
fargv = malloc(sizeof *fargv);
fargv->prev = NULL;
*fargv = malloc(sizeof **fargv);
(*fargv)->prev = NULL;
}

{
Expand All @@ -2752,18 +2755,23 @@ dofunction(Char **v, struct command *t)
}

vh[i] = NULL;
fargv->v = vh;
(*fargv)->v = vh;
}
fcurr.ready = 1;

srcfile(short2str(ffile), 0, 0, NULL);
if (!srcfile(file = fcurr.file ? fcurr.file :
strsave(short2str(ffile)), 0, 0, NULL))
stderror(ERR_SYSTEM, file, strerror(errno));
if (file != fcurr.file)
xfree(file);
/* Reset STRargv on function exit. */
setv(STRargv, NULL, VAR_READWRITE);

if (fargv->prev) {
fargv = fargv->prev;
free(fargv->next);
if ((*fargv)->prev) {
*fargv = (*fargv)->prev;
free((*fargv)->next);
} else {
free(fargv);
fargv = NULL;
free(*fargv);
*fargv = NULL;
}
}
22 changes: 16 additions & 6 deletions sh.h
Original file line number Diff line number Diff line change
Expand Up @@ -1308,12 +1308,22 @@ extern int filec;
/* Function variable(s) and function(s). */
extern Char *Sgoal;
extern int Stype;
extern struct funcargs {
Char **v;
int eof;
struct funcargs *prev,
*next;
} *fargv;
extern struct funccurr {
struct funcargs {
Char **v;
struct funcargs *prev,
*next;
} *fargv;
struct funcfile {
char *file;
struct funcfile *prev,
*next;
} *ffile;
char *file;
int eof,
src,
ready;
} fcurr;
extern int getword(struct Strbuf *);
extern int srcfile(const char *, int, int, Char **);

Expand Down