Commit 3ecfb738557e00b5dfd1d6bddabcdb2d113996e1
1 parent
492a4559
Exists in
master
and in
3 other branches
Refactoring timer engine to avoid an abnormal delay when calling
wait_for_ready on windows.
Showing
4 changed files
with
74 additions
and
169 deletions
Show diff stats
src/core/ctlr.c
... | ... | @@ -2696,101 +2696,3 @@ enum dbcs_state ctlr_dbcs_state(int baddr) |
2696 | 2696 | return dbcs? ea_buf[baddr].db: DBCS_NONE; |
2697 | 2697 | } |
2698 | 2698 | #endif /*]*/ |
2699 | - | |
2700 | -/* | |
2701 | - * Transaction timing. The time between sending an interrupt (PF, PA, Enter, | |
2702 | - * Clear) and the host unlocking the keyboard is indicated on the status line | |
2703 | - * to an accuracy of 0.1 seconds. If we don't repaint the screen before we see | |
2704 | - * the unlock, the time should be fairly accurate. | |
2705 | - */ | |
2706 | -//static struct timeval t_start; | |
2707 | -// static Boolean ticking = False; | |
2708 | -//static Boolean mticking = False; | |
2709 | -//static void * tick_id; | |
2710 | -//static struct timeval t_want; | |
2711 | - | |
2712 | -/* | |
2713 | -/// @brief Return the difference in milliseconds between two timevals. | |
2714 | -static long delta_msec(struct timeval *t1, struct timeval *t0) | |
2715 | -{ | |
2716 | - return (t1->tv_sec - t0->tv_sec) * 1000 + | |
2717 | - (t1->tv_usec - t0->tv_usec + 500) / 1000; | |
2718 | -} | |
2719 | - | |
2720 | -static int keep_ticking(H3270 *hSession) | |
2721 | -{ | |
2722 | - struct timeval t1; | |
2723 | - long msec; | |
2724 | - | |
2725 | - do | |
2726 | - { | |
2727 | - (void) gettimeofday(&t1, (struct timezone *) 0); | |
2728 | - hSession->t_want.tv_sec++; | |
2729 | - msec = delta_msec(&hSession->t_want, &t1); | |
2730 | - } while (msec <= 0); | |
2731 | - | |
2732 | - status_timing(hSession,&hSession->t_start, &t1); | |
2733 | - | |
2734 | - return 1; | |
2735 | -} | |
2736 | -*/ | |
2737 | - | |
2738 | -/* | |
2739 | -void ticking_start(H3270 *hSession, Boolean anyway) | |
2740 | -{ | |
2741 | - CHECK_SESSION_HANDLE(hSession); | |
2742 | - | |
2743 | - if(hSession->cbk.set_timer) | |
2744 | - { | |
2745 | - if(lib3270_get_toggle(hSession,LIB3270_TOGGLE_SHOW_TIMING) || anyway) | |
2746 | - hSession->cbk.set_timer(hSession,1); | |
2747 | - } | |
2748 | - else | |
2749 | - { | |
2750 | - (void) gettimeofday(&hSession->t_start, (struct timezone *) 0); | |
2751 | - | |
2752 | - hSession->mticking = 1; | |
2753 | - | |
2754 | - if (!lib3270_get_toggle(hSession,LIB3270_TOGGLE_SHOW_TIMING) && !anyway) | |
2755 | - return; | |
2756 | - | |
2757 | - status_untiming(hSession); | |
2758 | - | |
2759 | - if (hSession->ticking) | |
2760 | - RemoveTimer(hSession, hSession->tick_id); | |
2761 | - | |
2762 | - hSession->ticking = 1; | |
2763 | - hSession->tick_id = AddTimer(1000, hSession, keep_ticking); | |
2764 | - hSession->t_want = hSession->t_start; | |
2765 | - } | |
2766 | - | |
2767 | -} | |
2768 | -*/ | |
2769 | - | |
2770 | -/* | |
2771 | -static void ticking_stop(H3270 *hSession) | |
2772 | -{ | |
2773 | - CHECK_SESSION_HANDLE(hSession); | |
2774 | - | |
2775 | - if(hSession->cbk.set_timer) | |
2776 | - { | |
2777 | - hSession->cbk.set_timer(hSession,0); | |
2778 | - } | |
2779 | - else | |
2780 | - { | |
2781 | - struct timeval t1; | |
2782 | - | |
2783 | - (void) gettimeofday(&t1, (struct timezone *) 0); | |
2784 | - if (hSession->mticking) | |
2785 | - hSession->mticking = 0; | |
2786 | - else | |
2787 | - return; | |
2788 | - | |
2789 | - if (!hSession->ticking) | |
2790 | - return; | |
2791 | - RemoveTimer(hSession, hSession->tick_id); | |
2792 | - hSession->ticking = 0; | |
2793 | - status_timing(hSession,&hSession->t_start, &t1); | |
2794 | - } | |
2795 | -} | |
2796 | -*/ | ... | ... |
src/core/linux/event_dispatcher.c
... | ... | @@ -188,10 +188,14 @@ retry: |
188 | 188 | if (t->tv.tv_sec < now.tv_sec ||(t->tv.tv_sec == now.tv_sec && t->tv.tv_usec < now.tv_usec)) |
189 | 189 | { |
190 | 190 | t->in_play = True; |
191 | - (*t->proc)(hSession,t->userdata); | |
191 | + | |
192 | + if((*t->proc)(hSession,t->userdata) == 0) | |
193 | + lib3270_linked_list_delete_node(&hSession->timeouts,t); | |
194 | + else | |
195 | + t->in_play = False; | |
196 | + | |
192 | 197 | processed_any = True; |
193 | 198 | |
194 | - lib3270_linked_list_delete_node(&hSession->timeouts,t); | |
195 | 199 | |
196 | 200 | } |
197 | 201 | else | ... | ... |
src/core/wait.c
... | ... | @@ -31,9 +31,16 @@ |
31 | 31 | #include <lib3270/log.h> |
32 | 32 | #include <lib3270/trace.h> |
33 | 33 | #include "kybdc.h" |
34 | +#include "utilc.h" | |
34 | 35 | |
35 | 36 | /*---[ Implement ]------------------------------------------------------------------------------------------*/ |
36 | 37 | |
38 | +static int timer_expired(H3270 GNUC_UNUSED(*hSession), void *userdata) | |
39 | +{ | |
40 | + *((int *) userdata) = errno = ETIMEDOUT; | |
41 | + return 1; // Keep timer handle. | |
42 | +} | |
43 | + | |
37 | 44 | LIB3270_EXPORT int lib3270_wait_for_update(H3270 GNUC_UNUSED(*hSession), int GNUC_UNUSED(seconds)) |
38 | 45 | { |
39 | 46 | return errno = ENOTSUP; |
... | ... | @@ -41,57 +48,70 @@ LIB3270_EXPORT int lib3270_wait_for_update(H3270 GNUC_UNUSED(*hSession), int GNU |
41 | 48 | |
42 | 49 | LIB3270_EXPORT int lib3270_wait_for_ready(H3270 *hSession, int seconds) |
43 | 50 | { |
44 | - time_t end = time(0)+seconds; | |
45 | - | |
46 | - if(lib3270_is_disconnected(hSession)) | |
47 | - return errno = ENOTCONN; | |
48 | - | |
49 | - lib3270_main_iterate(hSession,0); | |
51 | + FAIL_IF_NOT_ONLINE(hSession); | |
50 | 52 | |
51 | - // Keyboard is locked by operator error, fails! | |
52 | - if(hSession->kybdlock && KYBDLOCK_IS_OERR(hSession)) | |
53 | - return errno = EPERM; | |
53 | + int rc = 0; | |
54 | + void * timer = AddTimer(seconds * 1000, hSession, timer_expired, &rc); | |
54 | 55 | |
55 | - do | |
56 | + while(!rc) | |
56 | 57 | { |
57 | 58 | if(!lib3270_get_lock_status(hSession)) |
58 | - return 0; | |
59 | + { | |
60 | + break; | |
61 | + } | |
59 | 62 | |
60 | 63 | if(lib3270_is_disconnected(hSession)) |
61 | - return errno = ENOTCONN; | |
64 | + { | |
65 | + rc = errno = ENOTCONN; | |
66 | + break; | |
67 | + } | |
62 | 68 | |
63 | - lib3270_main_iterate(hSession,1); | |
69 | + if(hSession->kybdlock && KYBDLOCK_IS_OERR(hSession)) | |
70 | + { | |
71 | + rc = errno = EPERM; | |
72 | + break; | |
73 | + } | |
64 | 74 | |
75 | + lib3270_main_iterate(hSession,1); | |
65 | 76 | } |
66 | - while(time(0) < end); | |
77 | + RemoveTimer(hSession,timer); | |
78 | + | |
79 | + return rc; | |
67 | 80 | |
68 | - return errno = ETIMEDOUT; | |
69 | 81 | } |
70 | 82 | |
71 | 83 | int lib3270_wait_for_string(H3270 *hSession, const char *key, int seconds) |
72 | 84 | { |
73 | - time_t end = time(0)+seconds; | |
74 | - | |
75 | 85 | FAIL_IF_NOT_ONLINE(hSession); |
76 | 86 | |
77 | - lib3270_main_iterate(hSession,0); | |
87 | + int rc = 0; | |
88 | + void * timer = AddTimer(seconds * 1000, hSession, timer_expired, &rc); | |
78 | 89 | |
79 | - do | |
90 | + while(!rc) | |
80 | 91 | { |
81 | 92 | // Keyboard is locked by operator error, fails! |
82 | 93 | if(hSession->kybdlock && KYBDLOCK_IS_OERR(hSession)) |
83 | - return errno = EPERM; | |
94 | + { | |
95 | + rc = errno = EPERM; | |
96 | + break; | |
97 | + } | |
84 | 98 | |
85 | 99 | if(!lib3270_is_connected(hSession)) |
86 | - return errno = ENOTCONN; | |
100 | + { | |
101 | + rc = errno = ENOTCONN; | |
102 | + break; | |
103 | + } | |
87 | 104 | |
88 | 105 | char * contents = lib3270_get_string_at_address(hSession, 0, -1, 0); |
89 | 106 | if(!contents) |
90 | - return errno; | |
107 | + { | |
108 | + rc = errno; | |
109 | + break; | |
110 | + } | |
91 | 111 | |
92 | 112 | if(strstr(contents,key)) { |
93 | 113 | lib3270_free(contents); |
94 | - return 0; | |
114 | + break; | |
95 | 115 | } |
96 | 116 | |
97 | 117 | lib3270_free(contents); |
... | ... | @@ -99,40 +119,48 @@ int lib3270_wait_for_string(H3270 *hSession, const char *key, int seconds) |
99 | 119 | lib3270_main_iterate(hSession,1); |
100 | 120 | |
101 | 121 | } |
102 | - while(time(0) < end); | |
122 | + RemoveTimer(hSession,timer); | |
123 | + | |
124 | + return rc; | |
103 | 125 | |
104 | - return errno = ETIMEDOUT; | |
105 | 126 | } |
106 | 127 | |
107 | 128 | int lib3270_wait_for_string_at_address(H3270 *hSession, int baddr, const char *key, int seconds) |
108 | 129 | { |
109 | - time_t end = time(0)+seconds; | |
110 | - | |
111 | 130 | FAIL_IF_NOT_ONLINE(hSession); |
112 | 131 | |
113 | - lib3270_main_iterate(hSession,0); | |
114 | - | |
115 | 132 | if(baddr < 0) |
116 | 133 | baddr = lib3270_get_cursor_address(hSession); |
117 | 134 | |
118 | - do | |
135 | + int rc = 0; | |
136 | + void * timer = AddTimer(seconds * 1000, hSession, timer_expired, &rc); | |
137 | + | |
138 | + while(!rc) | |
119 | 139 | { |
120 | 140 | // Keyboard is locked by operator error, fails! |
121 | 141 | if(hSession->kybdlock && KYBDLOCK_IS_OERR(hSession)) |
122 | - return errno = EPERM; | |
142 | + { | |
143 | + rc = errno = EPERM; | |
144 | + break; | |
145 | + } | |
123 | 146 | |
124 | 147 | if(!lib3270_is_connected(hSession)) |
125 | - return errno = ENOTCONN; | |
148 | + { | |
149 | + rc = errno = ENOTCONN; | |
150 | + break; | |
151 | + } | |
126 | 152 | |
127 | 153 | if(lib3270_cmp_string_at_address(hSession, baddr, key, 0) == 0) |
128 | - return 0; | |
154 | + { | |
155 | + break; | |
156 | + } | |
129 | 157 | |
130 | 158 | lib3270_main_iterate(hSession,1); |
131 | 159 | |
132 | 160 | } |
133 | - while(time(0) < end); | |
161 | + RemoveTimer(hSession,timer); | |
134 | 162 | |
135 | - return errno = ETIMEDOUT; | |
163 | + return rc; | |
136 | 164 | |
137 | 165 | } |
138 | 166 | ... | ... |
src/include/utilc.h
... | ... | @@ -17,22 +17,11 @@ |
17 | 17 | * @brief Global declarations for util.c. |
18 | 18 | */ |
19 | 19 | |
20 | -LIB3270_INTERNAL char *ctl_see(int c); | |
20 | +LIB3270_INTERNAL char * ctl_see(int c); | |
21 | 21 | |
22 | - /* | |
23 | -LIB3270_INTERNAL void add_resource(const char *name, const char *value); | |
24 | -LIB3270_INTERNAL const char *get_message(const char *key); | |
25 | -LIB3270_INTERNAL const char *get_fresource(H3270 *hSession, const char *fmt, ...) LIB3270_GNUC_FORMAT(2, 3); | |
26 | -LIB3270_INTERNAL const char *get_resource(H3270 *hSession, const char *name); | |
27 | -LIB3270_INTERNAL int split_dbcs_resource(const char *value, char sep, char **part1, char **part2); | |
28 | -LIB3270_INTERNAL int split_dresource(char **st, char **left, char **right); | |
29 | -LIB3270_INTERNAL int split_lresource(char **st, char **value); | |
30 | -LIB3270_INTERNAL char *strip_whitespace(const char *s); | |
31 | -*/ | |
32 | - | |
33 | -LIB3270_INTERNAL char *xs_buffer(const char *fmt, ...) LIB3270_GNUC_FORMAT(1, 2); | |
34 | -LIB3270_INTERNAL void xs_error(const char *fmt, ...) LIB3270_GNUC_FORMAT(1, 2); | |
35 | -LIB3270_INTERNAL void xs_warning(const char *fmt, ...) LIB3270_GNUC_FORMAT(1, 2); | |
22 | +LIB3270_INTERNAL char * xs_buffer(const char *fmt, ...) LIB3270_GNUC_FORMAT(1, 2); | |
23 | +LIB3270_INTERNAL void xs_error(const char *fmt, ...) LIB3270_GNUC_FORMAT(1, 2); | |
24 | +LIB3270_INTERNAL void xs_warning(const char *fmt, ...) LIB3270_GNUC_FORMAT(1, 2); | |
36 | 25 | |
37 | 26 | LIB3270_INTERNAL void * AddInput(int, H3270 *session, void (*fn)(H3270 *session)); |
38 | 27 | LIB3270_INTERNAL void * AddOutput(int, H3270 *session, void (*fn)(H3270 *session)); |
... | ... | @@ -42,24 +31,6 @@ LIB3270_INTERNAL void RemoveSource(H3270 *session, void *cookie); |
42 | 31 | LIB3270_INTERNAL void * AddTimer(unsigned long msec, H3270 *session, int (*fn)(H3270 *session, void *userdata), void *userdata); |
43 | 32 | LIB3270_INTERNAL void RemoveTimer(H3270 *session, void *cookie); |
44 | 33 | |
45 | -// LIB3270_INTERNAL const char * KeysymToString(KeySym k); | |
46 | - | |
47 | -// LIB3270_INTERNAL int read_resource_file(const char *filename, Boolean fatal); | |
48 | -// LIB3270_INTERNAL Boolean split_hier(char *label, char **base, char ***parents); | |
49 | - | |
50 | -/* | |
51 | -typedef struct { | |
52 | - char *buf; | |
53 | - int alloc_len; | |
54 | - int cur_len; | |
55 | -} rpf_t; | |
56 | - | |
57 | -LIB3270_INTERNAL void rpf_init(rpf_t *r); | |
58 | -LIB3270_INTERNAL void rpf_reset(rpf_t *r); | |
59 | -LIB3270_INTERNAL void rpf(rpf_t *r, char *fmt, ...) LIB3270_GNUC_FORMAT(2, 3); | |
60 | -LIB3270_INTERNAL void rpf_free(rpf_t *r); | |
61 | -*/ | |
62 | - | |
63 | 34 | /** |
64 | 35 | * @brief "unescape" text (Replaces %value for corresponding character). |
65 | 36 | * |
... | ... | @@ -71,7 +42,7 @@ LIB3270_INTERNAL void rpf_free(rpf_t *r); |
71 | 42 | LIB3270_INTERNAL char * lib3270_unescape(const char *text); |
72 | 43 | |
73 | 44 | /** |
74 | - * @brief Compare strings ignoring non alfanumeric chars. | |
45 | + * @brief Compare strings ignoring non alphanumeric chars. | |
75 | 46 | * |
76 | 47 | * @param s1 First string. |
77 | 48 | * @param s2 Second string. | ... | ... |