aboutsummaryrefslogtreecommitdiff
path: root/shared/toxy/scriptlet.c
diff options
context:
space:
mode:
Diffstat (limited to 'shared/toxy/scriptlet.c')
-rw-r--r--shared/toxy/scriptlet.c224
1 files changed, 214 insertions, 10 deletions
diff --git a/shared/toxy/scriptlet.c b/shared/toxy/scriptlet.c
index 7d5146c..cd12f06 100644
--- a/shared/toxy/scriptlet.c
+++ b/shared/toxy/scriptlet.c
@@ -29,6 +29,16 @@
enum { SCRIPTLET_CVOK, SCRIPTLET_CVUNKNOWN, SCRIPTLET_CVMISSING };
+#define VERSLET_MAXPACKAGE 32
+#define VERSLET_MAXVERSION 32
+
+typedef struct _verslet
+{
+ t_pd *v_owner;
+ char v_package[VERSLET_MAXPACKAGE];
+ char v_version[VERSLET_MAXVERSION];
+} t_verslet;
+
struct _scriptlet
{
t_pd *s_owner;
@@ -618,12 +628,165 @@ char *scriptlet_nextword(char *buf)
return (0);
}
+static t_verslet *verslet_new(t_pd *owner)
+{
+ t_verslet *vp = getbytes(sizeof(*vp));
+ vp->v_owner = owner;
+ vp->v_package[0] = 0;
+ vp->v_version[0] = 0;
+ return (vp);
+}
+
+static void verslet_free(t_verslet *vp)
+{
+ freebytes(vp, sizeof(*vp));
+}
+
+static void verslet_set(t_verslet *vp, char *pname, char *vname)
+{
+ strncpy(vp->v_package, pname, VERSLET_MAXPACKAGE-1);
+ vp->v_package[VERSLET_MAXPACKAGE-1] = 0;
+ strncpy(vp->v_version, vname, VERSLET_MAXVERSION-1);
+ vp->v_version[VERSLET_MAXVERSION-1] = 0;
+}
+
+static int verslet_parse(t_verslet *vp, char *buf, int multiline)
+{
+ char *ptr = buf;
+ int plen = 0;
+ vp->v_package[0] = 0;
+ vp->v_version[0] = 0;
+ if (multiline)
+ {
+ while (*ptr)
+ {
+ while (*ptr == ' ' || *ptr == '\t') ptr++;
+ if (strncmp(ptr, "package", 7))
+ {
+ while (*ptr && *ptr != '\n') ptr++;
+ if (*ptr)
+ buf = ++ptr;
+ }
+ else break;
+ }
+ if (*ptr)
+ ptr += 7;
+ else
+ ptr = 0;
+ }
+ else
+ {
+ while (*ptr == ' ' || *ptr == '\t') ptr++;
+ if (strncmp(ptr, "package", 7))
+ ptr = 0;
+ else
+ ptr += 7;
+ }
+ if (ptr)
+ {
+ while (*ptr == ' ' || *ptr == '\t') ptr++;
+ if (!strncmp(ptr, "provide", 7))
+ {
+ ptr += 7;
+ while (*ptr == ' ' || *ptr == '\t') ptr++;
+ if (*ptr)
+ {
+ for (plen = 0; plen < VERSLET_MAXPACKAGE-1 && *ptr;
+ plen++, ptr++)
+ {
+ if (*ptr == '\n' || *ptr == '\r')
+ break;
+ else if (*ptr == ' ' || *ptr == '\t')
+ {
+ vp->v_package[plen] = 0;
+#ifdef SCRIPTLET_DEBUG
+ fprintf(stderr, "package \"%s\"\n", vp->v_package);
+#endif
+ while (*ptr == ' ' || *ptr == '\t') ptr++;
+ if (*ptr >= '0' && *ptr <= '9')
+ {
+ int vlen;
+ for (vlen = 0; vlen < VERSLET_MAXVERSION-1 && *ptr;
+ vlen++, ptr++)
+ {
+ if ((*ptr >= '0' && *ptr <= '9') || *ptr == '.')
+ vp->v_version[vlen] = *ptr;
+ else
+ break;
+ }
+ if (vlen)
+ {
+ vp->v_version[vlen] = 0;
+#ifdef SCRIPTLET_DEBUG
+ fprintf(stderr, "version \"%s\"\n",
+ vp->v_version);
+#endif
+ return (1);
+ }
+ }
+ break;
+ }
+ else vp->v_package[plen] = *ptr;
+ }
+ }
+ }
+ if (plen)
+ loud_error(vp->v_owner,
+ "incomplete scriptlet version declaration \"%s\"", buf);
+ }
+ return (0);
+}
+
+static int verslet_compare(t_verslet *vp1, t_verslet *vp2)
+{
+ char *vname1 = vp1->v_version, *vname2 = vp2->v_version;
+ while (*vname1 || *vname2)
+ {
+ int v1, v2;
+ for (v1 = 0; *vname1 >= '0' && *vname1 <= '9'; vname1++)
+ v1 = v1 * 10 + *vname1 - '0';
+ for (v2 = 0; *vname2 >= '0' && *vname2 <= '9'; vname2++)
+ v2 = v2 * 10 + *vname2 - '0';
+ if (v1 < v2)
+ return (-1);
+ else if (v1 > v2)
+ return (1);
+ if (*vname1)
+ {
+ if (*vname1 == '.')
+ *vname1++;
+ if (*vname1 < '0' || *vname1 > '9')
+ {
+ loud_error(vp1->v_owner, "invalid version \"%s\"",
+ vp1->v_version);
+ while (*vname1) *vname1++;
+ }
+ }
+ if (*vname2)
+ {
+ if (*vname2 == '.')
+ *vname2++;
+ if (*vname2 < '0' || *vname2 > '9')
+ {
+ loud_error(vp2->v_owner, "invalid version \"%s\"",
+ vp2->v_version);
+ while (*vname2) *vname2++;
+ }
+ }
+ }
+ return (0);
+}
+
static int scriptlet_doread(t_scriptlet *sp, t_pd *caller, FILE *fp,
- char *rc, char *builtin, t_scriptlet_cmntfn cmntfn)
+ char *rc, t_verslet *vcompare,
+ char *builtin, t_scriptlet_cmntfn cmntfn)
{
t_scriptlet *outsp = sp, *newsp;
+ t_verslet *vp;
+ int vdiff = 0;
char buf[MAXPDSTRING];
if (!caller) caller = sp->s_owner;
+ vp = (vcompare ? verslet_new(caller) : 0);
while ((fp && !feof(fp) && fgets(buf, MAXPDSTRING - 1, fp))
|| builtin)
{
@@ -643,9 +806,22 @@ static int scriptlet_doread(t_scriptlet *sp, t_pd *caller, FILE *fp,
else builtin++;
}
}
- else for (ptr = buf; *ptr; ptr++)
- if (*ptr == '\r')
- *ptr = ' '; /* LATER rethink */
+ else
+ {
+ for (ptr = buf; *ptr; ptr++)
+ if (*ptr == '\r')
+ *ptr = ' '; /* LATER rethink */
+ if (vp && verslet_parse(vp, buf, 0))
+ {
+ if (vdiff = verslet_compare(vp, vcompare))
+ goto readfailed;
+ else
+ {
+ verslet_free(vp);
+ vp = 0;
+ }
+ }
+ }
ptr = buf;
while (*ptr == ' ' || *ptr == '\t') ptr++;
if (*ptr == '#')
@@ -666,6 +842,8 @@ static int scriptlet_doread(t_scriptlet *sp, t_pd *caller, FILE *fp,
ep--;
ep[1] = 0;
}
+ if (vp)
+ goto readfailed; /* FIXME call a request cmntfn? */
newsp = cmntfn(caller, rc, sel, ptr);
if (newsp == SCRIPTLET_UNLOCK)
outsp->s_locked = 0;
@@ -682,8 +860,20 @@ static int scriptlet_doread(t_scriptlet *sp, t_pd *caller, FILE *fp,
else if (*ptr && *ptr != '\n')
scriptlet_doappend(outsp, buf);
}
+readfailed:
outsp->s_locked = 0;
- return (SCRIPTLET_OK);
+ if (vp)
+ {
+ verslet_free(vp);
+ scriptlet_reset(sp);
+ if (vdiff < 0)
+ return (SCRIPTLET_OLDERVERSION);
+ else if (vdiff > 0)
+ return (SCRIPTLET_NEWERVERSION);
+ else
+ return (SCRIPTLET_NOVERSION);
+ }
+ else return (SCRIPTLET_OK);
}
/* Load particular section(s) from buffer (skip up to an unlocking comment,
@@ -693,7 +883,7 @@ int scriptlet_rcparse(t_scriptlet *sp, t_pd *caller, char *rc, char *contents,
{
int result;
sp->s_locked = 1; /* see scriptlet_doread() above for unlocking scheme */
- result = scriptlet_doread(sp, caller, 0, rc, contents, cmntfn);
+ result = scriptlet_doread(sp, caller, 0, rc, 0, contents, cmntfn);
return (result);
}
@@ -725,18 +915,32 @@ int scriptlet_rcload(t_scriptlet *sp, t_pd *caller, char *rc, char *ext,
else sys_bashfilename(nameptr, filename);
if (fp = fopen(filename, "r"))
{
- result = scriptlet_doread(sp, caller, fp, rc, 0, cmntfn);
+ t_verslet *vp;
+ if (builtin)
+ {
+ vp = verslet_new(sp->s_owner);
+ if (!verslet_parse(vp, builtin, 1))
+ {
+ bug("scriptlet_rcload 1");
+ verslet_free(vp);
+ vp = 0;
+ }
+ }
+ else vp = 0;
+ result = scriptlet_doread(sp, caller, fp, rc, vp, 0, cmntfn);
fclose(fp);
+ if (vp)
+ verslet_free(vp);
}
else
{
- bug("scriptlet_rcload");
+ bug("scriptlet_rcload 2");
result = SCRIPTLET_NOFILE;
}
}
if (result != SCRIPTLET_OK)
{
- scriptlet_doread(sp, caller, 0, rc, builtin, cmntfn);
+ scriptlet_doread(sp, caller, 0, rc, 0, builtin, cmntfn);
}
return (result);
}
@@ -756,7 +960,7 @@ int scriptlet_read(t_scriptlet *sp, t_symbol *fn)
if (fp = fopen(buf, "r"))
{
scriptlet_reset(sp);
- result = scriptlet_doread(sp, 0, fp, 0, 0, 0);
+ result = scriptlet_doread(sp, 0, fp, 0, 0, 0, 0);
fclose(fp);
}
else