-/* ptmacosx.c -- portable timer implementation for mac os x */\r
-\r
-#include <stdlib.h>\r
-#include <stdio.h>\r
-#include <CoreAudio/HostTime.h>\r
-\r
-#import <mach/mach.h>\r
-#import <mach/mach_error.h>\r
-#import <mach/mach_time.h>\r
-#import <mach/clock.h>\r
-#include <unistd.h>\r
-\r
-#include "porttime.h"\r
-#include "sys/time.h"\r
-#include "pthread.h"\r
-\r
-#define NSEC_PER_MSEC 1000000\r
-#define THREAD_IMPORTANCE 30\r
-\r
-static int time_started_flag = FALSE;\r
-static UInt64 start_time;\r
-static pthread_t pt_thread_pid;\r
-\r
-/* note that this is static data -- we only need one copy */\r
-typedef struct {\r
- int id;\r
- int resolution;\r
- PtCallback *callback;\r
- void *userData;\r
-} pt_callback_parameters;\r
-\r
-static int pt_callback_proc_id = 0;\r
-\r
-static void *Pt_CallbackProc(void *p)\r
-{\r
- pt_callback_parameters *parameters = (pt_callback_parameters *) p;\r
- int mytime = 1;\r
-\r
- kern_return_t error;\r
- thread_extended_policy_data_t extendedPolicy;\r
- thread_precedence_policy_data_t precedencePolicy;\r
-\r
- extendedPolicy.timeshare = 0;\r
- error = thread_policy_set(mach_thread_self(), THREAD_EXTENDED_POLICY,\r
- (thread_policy_t)&extendedPolicy,\r
- THREAD_EXTENDED_POLICY_COUNT);\r
- if (error != KERN_SUCCESS) {\r
- mach_error("Couldn't set thread timeshare policy", error);\r
- }\r
-\r
- precedencePolicy.importance = THREAD_IMPORTANCE;\r
- error = thread_policy_set(mach_thread_self(), THREAD_PRECEDENCE_POLICY,\r
- (thread_policy_t)&precedencePolicy,\r
- THREAD_PRECEDENCE_POLICY_COUNT);\r
- if (error != KERN_SUCCESS) {\r
- mach_error("Couldn't set thread precedence policy", error);\r
- }\r
- \r
- \r
- /* to kill a process, just increment the pt_callback_proc_id */\r
- /* printf("pt_callback_proc_id %d, id %d\n", pt_callback_proc_id, parameters->id); */\r
- while (pt_callback_proc_id == parameters->id) {\r
- /* wait for a multiple of resolution ms */\r
- UInt64 wait_time;\r
- int delay = mytime++ * parameters->resolution - Pt_Time();\r
- PtTimestamp timestamp;\r
- if (delay < 0) delay = 0;\r
- wait_time = AudioConvertNanosToHostTime((UInt64)delay * NSEC_PER_MSEC);\r
- wait_time += AudioGetCurrentHostTime();\r
- error = mach_wait_until(wait_time);\r
- timestamp = Pt_Time();\r
- (*(parameters->callback))(timestamp, parameters->userData);\r
- }\r
- free(parameters);\r
- return NULL;\r
-}\r
-\r
-\r
-PtError Pt_Start(int resolution, PtCallback *callback, void *userData)\r
-{\r
- if (time_started_flag) return ptAlreadyStarted;\r
- start_time = AudioGetCurrentHostTime();\r
- \r
- if (callback) {\r
- int res;\r
- pt_callback_parameters *parms;\r
-\r
- parms = (pt_callback_parameters *) malloc(sizeof(pt_callback_parameters));\r
- if (!parms) return ptInsufficientMemory;\r
- parms->id = pt_callback_proc_id;\r
- parms->resolution = resolution;\r
- parms->callback = callback;\r
- parms->userData = userData;\r
- res = pthread_create(&pt_thread_pid, NULL, Pt_CallbackProc, parms);\r
- if (res != 0) return ptHostError;\r
- }\r
- \r
- time_started_flag = TRUE;\r
- return ptNoError;\r
-}\r
-\r
-\r
-PtError Pt_Stop()\r
-{\r
- /* printf("Pt_Stop called\n"); */\r
- pt_callback_proc_id++;\r
- pthread_join(pt_thread_pid, NULL);\r
- time_started_flag = FALSE;\r
- return ptNoError;\r
-}\r
-\r
-\r
-int Pt_Started()\r
-{\r
- return time_started_flag;\r
-}\r
-\r
-\r
-PtTimestamp Pt_Time()\r
-{\r
- UInt64 clock_time, nsec_time;\r
- clock_time = AudioGetCurrentHostTime() - start_time;\r
- nsec_time = AudioConvertHostTimeToNanos(clock_time);\r
- return (PtTimestamp)(nsec_time / NSEC_PER_MSEC);\r
-}\r
-\r
-\r
-void Pt_Sleep(int32_t duration)\r
-{\r
- usleep(duration * 1000);\r
-}\r
+/* ptmacosx.c -- portable timer implementation for mac os x */
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <CoreAudio/HostTime.h>
+
+#import <mach/mach.h>
+#import <mach/mach_error.h>
+#import <mach/mach_time.h>
+#import <mach/clock.h>
+#include <unistd.h>
+
+#include "porttime.h"
+#include "sys/time.h"
+#include "pthread.h"
+
+#define NSEC_PER_MSEC 1000000
+#define THREAD_IMPORTANCE 30
+
+static int time_started_flag = FALSE;
+static UInt64 start_time;
+static pthread_t pt_thread_pid;
+
+/* note that this is static data -- we only need one copy */
+typedef struct {
+ int id;
+ int resolution;
+ PtCallback *callback;
+ void *userData;
+} pt_callback_parameters;
+
+static int pt_callback_proc_id = 0;
+
+static void *Pt_CallbackProc(void *p)
+{
+ pt_callback_parameters *parameters = (pt_callback_parameters *) p;
+ int mytime = 1;
+
+ kern_return_t error;
+ thread_extended_policy_data_t extendedPolicy;
+ thread_precedence_policy_data_t precedencePolicy;
+
+ extendedPolicy.timeshare = 0;
+ error = thread_policy_set(mach_thread_self(), THREAD_EXTENDED_POLICY,
+ (thread_policy_t)&extendedPolicy,
+ THREAD_EXTENDED_POLICY_COUNT);
+ if (error != KERN_SUCCESS) {
+ mach_error("Couldn't set thread timeshare policy", error);
+ }
+
+ precedencePolicy.importance = THREAD_IMPORTANCE;
+ error = thread_policy_set(mach_thread_self(), THREAD_PRECEDENCE_POLICY,
+ (thread_policy_t)&precedencePolicy,
+ THREAD_PRECEDENCE_POLICY_COUNT);
+ if (error != KERN_SUCCESS) {
+ mach_error("Couldn't set thread precedence policy", error);
+ }
+
+
+ /* to kill a process, just increment the pt_callback_proc_id */
+ /* printf("pt_callback_proc_id %d, id %d\n", pt_callback_proc_id, parameters->id); */
+ while (pt_callback_proc_id == parameters->id) {
+ /* wait for a multiple of resolution ms */
+ UInt64 wait_time;
+ int delay = mytime++ * parameters->resolution - Pt_Time();
+ PtTimestamp timestamp;
+ if (delay < 0) delay = 0;
+ wait_time = AudioConvertNanosToHostTime((UInt64)delay * NSEC_PER_MSEC);
+ wait_time += AudioGetCurrentHostTime();
+ error = mach_wait_until(wait_time);
+ timestamp = Pt_Time();
+ (*(parameters->callback))(timestamp, parameters->userData);
+ }
+ free(parameters);
+ return NULL;
+}
+
+
+PtError Pt_Start(int resolution, PtCallback *callback, void *userData)
+{
+ if (time_started_flag) return ptAlreadyStarted;
+ start_time = AudioGetCurrentHostTime();
+
+ if (callback) {
+ int res;
+ pt_callback_parameters *parms;
+
+ parms = (pt_callback_parameters *) malloc(sizeof(pt_callback_parameters));
+ if (!parms) return ptInsufficientMemory;
+ parms->id = pt_callback_proc_id;
+ parms->resolution = resolution;
+ parms->callback = callback;
+ parms->userData = userData;
+ res = pthread_create(&pt_thread_pid, NULL, Pt_CallbackProc, parms);
+ if (res != 0) return ptHostError;
+ }
+
+ time_started_flag = TRUE;
+ return ptNoError;
+}
+
+
+PtError Pt_Stop()
+{
+ /* printf("Pt_Stop called\n"); */
+ pt_callback_proc_id++;
+ pthread_join(pt_thread_pid, NULL);
+ time_started_flag = FALSE;
+ return ptNoError;
+}
+
+
+int Pt_Started()
+{
+ return time_started_flag;
+}
+
+
+PtTimestamp Pt_Time()
+{
+ UInt64 clock_time, nsec_time;
+ clock_time = AudioGetCurrentHostTime() - start_time;
+ nsec_time = AudioConvertHostTimeToNanos(clock_time);
+ return (PtTimestamp)(nsec_time / NSEC_PER_MSEC);
+}
+
+
+void Pt_Sleep(int32_t duration)
+{
+ usleep(duration * 1000);
+}