diff --git a/pd/src/m_class.c b/pd/src/m_class.c index a9ed52ad9c48003b6d2f5d8aba198242f5288041..59043974402913503674b6dc1efaa33d94ca7da4 100644 --- a/pd/src/m_class.c +++ b/pd/src/m_class.c @@ -786,7 +786,7 @@ void pd_typedmess(t_pd *x, t_symbol *s, int argc, t_atom *argv) ap++; break; case A_FLOAT: - if (!argc) goto badarg; + if (!argc) goto badarg; /* falls through */ case A_DEFFLOAT: if (!argc) *dp = 0; else @@ -814,7 +814,7 @@ void pd_typedmess(t_pd *x, t_symbol *s, int argc, t_atom *argv) ap++; break; case A_SYMBOL: - if (!argc) goto badarg; + if (!argc) goto badarg; /* falls through */ case A_DEFSYM: if (!argc) *ap = (t_int)(&s_); else diff --git a/pd/src/m_sched.c b/pd/src/m_sched.c index 55c464c6e6f0f8ceb87c8e74cb174b83e38b53bb..e51d64038efa9ce0fe460236ef44c80f1c45559c 100644 --- a/pd/src/m_sched.c +++ b/pd/src/m_sched.c @@ -405,8 +405,6 @@ void sched_set_using_audio(int flag) post("sorry, can't turn off callbacks yet; restart Pd"); /* not right yet! */ - sys_time_per_dsp_tick = (TIMEUNITPERSECOND) * - ((double)sys_schedblocksize) / sys_dacsr; // XXXFIXME //sys_vgui("pdtk_pd_dsp %s\n", flag ? "on" : "off"); } @@ -414,7 +412,8 @@ void sched_set_using_audio(int flag) /* take the scheduler forward one DSP tick, also handling clock timeouts */ void sched_tick( void) { - double next_sys_time = pd_this->pd_systime + sys_time_per_dsp_tick; + double next_sys_time = pd_this->pd_systime + + (sys_schedblocksize / sys_dacsr) * TIMEUNITPERSECOND; int countdown = 5000; while (pd_this->pd_clock_setlist && pd_this->pd_clock_setlist->c_settime < next_sys_time) @@ -459,6 +458,7 @@ int (*sys_idlehook)(void); static void m_pollingscheduler( void) { int idlecount = 0; + /* delete this when I'm sure it's not needed for back compatibilty? */ sys_time_per_dsp_tick = (TIMEUNITPERSECOND) * ((double)sys_schedblocksize) / sys_dacsr; @@ -635,8 +635,6 @@ int m_mainloop(void) int m_batchmain(void) { - sys_time_per_dsp_tick = (TIMEUNITPERSECOND) * - ((double)sys_schedblocksize) / sys_dacsr; while (sys_quit != SYS_QUIT_QUIT) sched_tick(); return (0); diff --git a/pd/src/s_inter.c b/pd/src/s_inter.c index 486ae7b0c18ac86e5c33c7511fc59891fe8d81a2..61bc9d94098fecc13c84e4c6064edefbd8798f87 100644 --- a/pd/src/s_inter.c +++ b/pd/src/s_inter.c @@ -291,12 +291,39 @@ void sys_setalarm(int microsec) #endif -#ifdef __linux +void sys_setsignalhandlers(void) +{ +#if !defined(_WIN32) && !defined(__CYGWIN__) + signal(SIGHUP, sys_huphandler); + signal(SIGINT, sys_exithandler); + signal(SIGQUIT, sys_exithandler); + signal(SIGILL, sys_exithandler); +# ifdef SIGIOT + signal(SIGIOT, sys_exithandler); +# endif + signal(SIGFPE, SIG_IGN); + /* signal(SIGILL, sys_exithandler); + signal(SIGBUS, sys_exithandler); + signal(SIGSEGV, sys_exithandler); */ + signal(SIGPIPE, SIG_IGN); + signal(SIGALRM, SIG_IGN); +#if 0 /* GG says: don't use that */ + signal(SIGSTKFLT, sys_exithandler); +#endif +#endif /* NOT _WIN32 && NOT __CYGWIN__ */ +} + +#if defined(__linux__) || defined(__FreeBSD_kernel__) || defined(__GNU__) + #if defined(_POSIX_PRIORITY_SCHEDULING) || defined(_POSIX_MEMLOCK) #include <sched.h> #endif +#define MODE_NRT 0 +#define MODE_RT 1 +#define MODE_WATCHDOG 2 + void sys_set_priority(int higher) { #ifdef _POSIX_PRIORITY_SCHEDULING @@ -1204,10 +1231,6 @@ int sys_startgui(const char *guidir) if (!getcwd(cwd, FILENAME_MAX)) strcpy(cwd, "."); #endif -#ifdef MSW - short version = MAKEWORD(2, 0); - WSADATA nobby; -#endif #ifdef HAVE_UNISTD_H int stdinpipe[2]; #endif @@ -1215,27 +1238,6 @@ int sys_startgui(const char *guidir) sys_fdpoll = (t_fdpoll *)t_getbytes(0); sys_nfdpoll = 0; inbinbuf = binbuf_new(); - -#ifdef HAVE_UNISTD_H - signal(SIGHUP, sys_huphandler); - signal(SIGINT, sys_exithandler); - signal(SIGQUIT, sys_exithandler); - signal(SIGILL, sys_exithandler); - signal(SIGIOT, sys_exithandler); - signal(SIGFPE, SIG_IGN); - /* signal(SIGILL, sys_exithandler); - signal(SIGBUS, sys_exithandler); - signal(SIGSEGV, sys_exithandler); */ - signal(SIGPIPE, SIG_IGN); - signal(SIGALRM, SIG_IGN); -#if 0 /* GG says: don't use that */ - signal(SIGSTKFLT, sys_exithandler); -#endif -#endif -#ifdef MSW - if (WSAStartup(version, &nobby)) sys_sockerror("WSAstartup"); -#endif - if (sys_nogui) { /* fake the GUI's message giving cwd and font sizes; then @@ -1374,7 +1376,6 @@ int sys_startgui(const char *guidir) /* this glob is needed so the Wish executable can have the same * filename as the Pd.app, i.e. 'Pd-0.42-3.app' should have a Wish * executable called 'Pd-0.42-3.app/Contents/MacOS/Pd-0.42-3' */ -// sprintf(embed_glob, "%s/../../MacOS/Pd*", guidir); sprintf(embed_glob, "%s/../../MacOS/nwjs", guidir); glob_buffer.gl_matchc = 1; /* we only need one match */ glob(embed_glob, GLOB_LIMIT, NULL, &glob_buffer); @@ -1408,7 +1409,7 @@ int sys_startgui(const char *guidir) use the nw binary and GUI code from your local copy of the Purr Data repo. (Make sure to run tar_em_up.sh first to fetch the nw binary.) */ - //strcpy(guidir2, "\"/home/grieg/purr-data/pd/nw\""); + //strcpy(guidir2, "\"/home/user/purr-data/pd/nw\""); sprintf(cmdbuf, "\"%s\" %s %s " "%d localhost %s %s " X_SPECIFIER, @@ -1449,7 +1450,7 @@ int sys_startgui(const char *guidir) use the nw binary and GUI code from your local copy of the Purr Data repo. (Make sure to run tar_em_up.sh first to fetch the nw binary.) */ - //strcpy(guidir2, "\"/home/grieg/purr-data/pd/nw\""); + //strcpy(guidir2, "\"/home/user/purr-data/pd/nw\""); sprintf(cmdbuf, "%s/nw/nw %s %s " "%d localhost %s %s " X_SPECIFIER, @@ -1472,11 +1473,15 @@ int sys_startgui(const char *guidir) { if (errno) perror("sys_startgui"); else fprintf(stderr, "sys_startgui failed\n"); + sys_closesocket(xsock); return (1); } else if (!childpid) /* we're the child */ { - setuid(getuid()); /* lose setuid priveliges */ + sys_closesocket(xsock); /* we're the child */ +#if defined(__linux__) || defined(__FreeBSD_kernel__) || defined(__GNU__) + sys_set_priority(MODE_NRT); +#endif #ifndef __APPLE__ /* the wish process in Unix will make a wish shell and read/write standard in and out unless we close the @@ -1548,23 +1553,101 @@ int sys_startgui(const char *guidir) #endif /* MSW */ } + if (!sys_nogui && !sys_guisetportnumber) + { + if (sys_verbose) + fprintf(stderr, "Waiting for connection request... \n"); + if (listen(xsock, 5) < 0) sys_sockerror("listen"); + + sys_guisock = accept(xsock, (struct sockaddr *) &server, + (socklen_t *)&len); +#ifdef OOPS + sys_closesocket(xsock); +#endif + if (sys_guisock < 0) sys_sockerror("accept"); + if (sys_verbose) + fprintf(stderr, "... connected\n"); + } + if (!sys_nogui) + { + char buf[256], buf2[256]; + sys_socketreceiver = socketreceiver_new(0, 0, 0, 0); + sys_addpollfn(sys_guisock, (t_fdpollfn)socketreceiver_read, + sys_socketreceiver); + + /* here is where we start the pinging. */ #if defined(__linux__) || defined(IRIX) - /* now that we've spun off the child process we can promote - our process's priority, if we can and want to. If not specfied - (-1), we assume real-time was wanted. Afterward, just in case - someone made Pd setuid in order to get permission to do this, - unset setuid and lose root priveliges after doing this. Starting - in Linux 2.6 this is accomplished by putting lines like: + if (sys_hipriority) + gui_vmess("gui_watchdog", ""); +#endif + sys_get_audio_apis(buf); + sys_get_midi_apis(buf2); + + t_binbuf *aapis = binbuf_new(), *mapis = binbuf_new(); + sys_get_audio_apis2(aapis); + sys_get_midi_apis2(mapis); + gui_start_vmess("gui_startup", "sss", + pd_version, + sys_font, + sys_fontweight); + + int i; + gui_start_array(); // audio apis + for (i = 0; i < binbuf_getnatom(aapis); i+=2) + { + gui_s(atom_getsymbol(binbuf_getvec(aapis)+i)->s_name); + gui_i(atom_getint(binbuf_getvec(aapis)+i+1)); + } + gui_end_array(); + + gui_start_array(); // midi apis + for (i = 0; i < binbuf_getnatom(mapis); i+=2) + { + gui_s(atom_getsymbol(binbuf_getvec(mapis)+i)->s_name); + gui_i(atom_getint(binbuf_getvec(mapis)+i+1)); + } + gui_end_array(); + + gui_end_vmess(); + gui_vmess("gui_set_cwd", "xs", 0, cwd); + binbuf_free(aapis); + binbuf_free(mapis); + } + return (0); + +} + +void sys_setrealtime(const char *libdir) +{ + char cmdbuf[MAXPDSTRING]; +#if defined(__linux__) || defined(__FreeBSD_kernel__) + /* promote this process's priority, if we can and want to. + If sys_hipriority not specfied (-1), we assume real-time was wanted. + Starting in Linux 2.6 one can permit real-time operation of Pd by + putting lines like: @audio - rtprio 99 @audio - memlock unlimited in the system limits file, perhaps /etc/limits.conf or - /etc/security/limits.conf */ + /etc/security/limits.conf, and calling Pd from a user in group audio. */ if (sys_hipriority == -1) - sys_hipriority = 1; //(!getuid() || !geteuid()); - - sprintf(cmdbuf, "%s/pd-watchdog\n", guidir); + sys_hipriority = 1; + + snprintf(cmdbuf, MAXPDSTRING, "%s/bin/pd-watchdog", libdir); + cmdbuf[MAXPDSTRING-1] = 0; + if (sys_hipriority) + { + struct stat statbuf; + if (stat(cmdbuf, &statbuf) < 0) + { + fprintf(stderr, + "disabling real-time priority due to missing pd-watchdog (%s)\n", + cmdbuf); + sys_hipriority = 0; + } + } if (sys_hipriority) { + int pipe9[2], watchpid; /* To prevent lockup, we fork off a watchdog process with higher real-time priority than ours. The GUI has to send a stream of ping messages to the watchdog THROUGH the Pd @@ -1574,27 +1657,23 @@ int sys_startgui(const char *guidir) to make it timeshare with the rest of the system. (Version 0.33P2 : if there's no GUI, the watchdog pinging is done from the scheduler idle routine in this process instead.) */ - int pipe9[2], watchpid; if (pipe(pipe9) < 0) { - setuid(getuid()); /* lose setuid priveliges */ sys_sockerror("pipe"); - return (1); + return; } watchpid = fork(); if (watchpid < 0) { - setuid(getuid()); /* lose setuid priveliges */ if (errno) - perror("sys_startgui"); - else fprintf(stderr, "sys_startgui failed\n"); - return (1); + perror("sys_setpriority"); + else fprintf(stderr, "sys_setpriority failed\n"); + return; } else if (!watchpid) /* we're the child */ { - sys_set_priority(1); - setuid(getuid()); /* lose setuid priveliges */ + sys_set_priority(MODE_WATCHDOG); if (pipe9[1] != 0) { dup2(pipe9[0], 0); @@ -1602,26 +1681,31 @@ int sys_startgui(const char *guidir) } close(pipe9[1]); - if (sys_verbose) fprintf(stderr, "%s", cmdbuf); + if (sys_verbose) fprintf(stderr, "%s\n", cmdbuf); execl("/bin/sh", "sh", "-c", cmdbuf, (char*)0); perror("pd: exec"); _exit(1); } else /* we're the parent */ { - sys_set_priority(0); - setuid(getuid()); /* lose setuid priveliges */ + sys_set_priority(MODE_RT); close(pipe9[0]); + /* set close-on-exec so that watchdog will see an EOF when we + close our copy - otherwise it might hang waiting for some + stupid child process (as seems to happen if jackd auto-starts + for us.) */ + if(fcntl(pipe9[1], F_SETFD, FD_CLOEXEC) < 0) + perror("close-on-exec"); sys_watchfd = pipe9[1]; /* We also have to start the ping loop in the GUI; this is done later when the socket is open. */ } } - - setuid(getuid()); /* lose setuid priveliges */ + else if (sys_verbose) + post("not setting real-time priority"); #endif /* __linux__ */ -#ifdef MSW +#ifdef _WIN32 if (!SetPriorityClass(GetCurrentProcess(), HIGH_PRIORITY_CLASS)) fprintf(stderr, "pd: couldn't set high priority class\n"); #endif @@ -1638,71 +1722,6 @@ int sys_startgui(const char *guidir) post("warning: high priority scheduling failed\n"); } #endif /* __APPLE__ */ - - if (!sys_nogui && !sys_guisetportnumber) - { - if (sys_verbose) - fprintf(stderr, "Waiting for connection request... \n"); - if (listen(xsock, 5) < 0) sys_sockerror("listen"); - - sys_guisock = accept(xsock, (struct sockaddr *) &server, - (socklen_t *)&len); -#ifdef OOPS - sys_closesocket(xsock); -#endif - if (sys_guisock < 0) sys_sockerror("accept"); - if (sys_verbose) - fprintf(stderr, "... connected\n"); - } - if (!sys_nogui) - { - char buf[256], buf2[256]; - sys_socketreceiver = socketreceiver_new(0, 0, 0, 0); - sys_addpollfn(sys_guisock, (t_fdpollfn)socketreceiver_read, - sys_socketreceiver); - - /* here is where we start the pinging. */ -#if defined(__linux__) || defined(IRIX) - if (sys_hipriority) - gui_vmess("gui_watchdog", ""); -#endif - sys_get_audio_apis(buf); - sys_get_midi_apis(buf2); -// sys_vgui("pdtk_pd_startup {%s} %s %s {%s} %s\n", pd_version, buf, buf2, -// sys_font, sys_fontweight); - - t_binbuf *aapis = binbuf_new(), *mapis = binbuf_new(); - sys_get_audio_apis2(aapis); - sys_get_midi_apis2(mapis); - gui_start_vmess("gui_startup", "sss", - pd_version, - sys_font, - sys_fontweight); - - int i; - gui_start_array(); // audio apis - for (i = 0; i < binbuf_getnatom(aapis); i+=2) - { - gui_s(atom_getsymbol(binbuf_getvec(aapis)+i)->s_name); - gui_i(atom_getint(binbuf_getvec(aapis)+i+1)); - } - gui_end_array(); - - gui_start_array(); // midi apis - for (i = 0; i < binbuf_getnatom(mapis); i+=2) - { - gui_s(atom_getsymbol(binbuf_getvec(mapis)+i)->s_name); - gui_i(atom_getint(binbuf_getvec(mapis)+i+1)); - } - gui_end_array(); - - gui_end_vmess(); - gui_vmess("gui_set_cwd", "xs", 0, cwd); - binbuf_free(aapis); - binbuf_free(mapis); - } - return (0); - } extern void sys_exit(void); diff --git a/pd/src/s_main.c b/pd/src/s_main.c index 18e384dc5a1ccf41169f40824ea2c1bb0051dfe1..dca6b1fd8a203695c6dd72c9f0644596b91edfaa 100644 --- a/pd/src/s_main.c +++ b/pd/src/s_main.c @@ -32,7 +32,9 @@ static const char pd_compiledate[] = __DATE__; void pd_init(void); int sys_argparse(int argc, char **argv); void sys_findprogdir(char *progname); +void sys_setsignalhandlers(void); int sys_startgui(const char *guipath); +void sys_setrealtime(const char *guipath); int sys_rcfile(void); int m_mainloop(void); int m_batchmain(void); @@ -243,6 +245,41 @@ void glob_initfromgui(void *dummy, t_symbol *s, int argc, t_atom *argv) sys_messagelist = 0; } + +// font char metric triples: pointsize width(pixels) height(pixels) +static int defaultfontshit[] = { + 8, 5, 11, 9, 6, 12, + 10, 6, 13, 12, 7, 16, + 14, 8, 17, 16, 10, 19, + 18, 11, 22, 24, 14, 29, + 30, 18, 37, 36, 22, 44 +}; +#define NDEFAULTFONT (sizeof(defaultfontshit)/sizeof(*defaultfontshit)) + +static t_clock *sys_fakefromguiclk; +static void sys_fakefromgui(void) +{ + /* fake the GUI's message giving cwd and font sizes in case + we aren't starting the gui. */ + t_atom zz[NDEFAULTFONT+2]; + int i; + char buf[MAXPDSTRING]; +#ifdef _WIN32 + if (GetCurrentDirectory(MAXPDSTRING, buf) == 0) + strcpy(buf, "."); +#else + if (!getcwd(buf, MAXPDSTRING)) + strcpy(buf, "."); + +#endif + SETSYMBOL(zz, gensym(buf)); + for (i = 0; i < (int)NDEFAULTFONT; i++) + SETFLOAT(zz+i+1, defaultfontshit[i]); + SETFLOAT(zz+NDEFAULTFONT+1,0); + glob_initfromgui(0, 0, 2+NDEFAULTFONT, zz); + clock_free(sys_fakefromguiclk); +} + static void sys_afterargparse(void); static void pd_makeversion(void) @@ -298,6 +335,34 @@ int sys_main(int argc, char **argv) #ifdef PD_DEBUG fprintf(stderr, "Pd-L2Ork: COMPILED FOR DEBUGGING\n"); #endif + /* We need to call WSAStartup regardless of gui mode, since a user + * might want to make socket connections even in -nogui mode. So we + * go ahead and do that here. */ +#ifdef _WIN32 + short version = MAKEWORD(2, 0); + WSADATA nobby; + if (WSAStartup(version, &nobby)) sys_sockerror("WSAstartup"); + /* use Win32 "binary" mode by default since we don't want the + * translation that Win32 does by default */ +# ifdef _MSC_VER /* MS Visual Studio */ + _set_fmode( _O_BINARY ); +# else /* MinGW */ + { + extern int _fmode; + _fmode = _O_BINARY; + } +# endif /* _MSC_VER */ +#endif /* _WIN32 */ +#ifndef _WIN32 + /* long ago Pd used setuid to promote itself to real-time priority. + Just in case anyone's installation script still makes it setuid, we + complain to stderr and lose setuid here. */ + if (getuid() != geteuid()) + { + fprintf(stderr, "warning: canceling setuid privilege\n"); + setuid(getuid()); + } +#endif /* _WIN32 */ pd_init(); /* start the message system */ logpost(NULL, 2, "PD_FLOATSIZE = %lu bits", sizeof(t_float)*8); sys_findprogdir(argv[0]); /* set sys_progname, guipath */ @@ -320,7 +385,11 @@ int sys_main(int argc, char **argv) pd_version, pd_compiletime, pd_compiledate); if (sys_version) /* if we were just asked our version, exit here. */ return (0); - if (sys_startgui(sys_guidir->s_name)) /* start the gui */ + sys_setsignalhandlers(); + if (sys_nogui) + clock_set((sys_fakefromguiclk = + clock_new(0, (t_method)sys_fakefromgui)), 0); + else if (sys_startgui(sys_guidir->s_name)) /* start the gui */ return(1); /* send the libdir to the GUI */ gui_vmess("gui_set_lib_dir", "s", sys_libdir->s_name); @@ -340,6 +409,8 @@ int sys_main(int argc, char **argv) gui_end_array(); gui_end_vmess(); + if (sys_hipriority) + sys_setrealtime(sys_libdir->s_name); /* set desired process priority */ if (sys_externalschedlib) return (sys_run_scheduler(sys_externalschedlibname, sys_extraflagsstring));