test_timers.cpp 7.13 KB
Newer Older
somdoron's avatar
somdoron committed
1
/*
2
    Copyright (c) 2007-2016 Contributors as noted in the AUTHORS file
somdoron's avatar
somdoron committed
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29

    This file is part of libzmq, the ZeroMQ core engine in C++.

    libzmq is free software; you can redistribute it and/or modify it under
    the terms of the GNU Lesser General Public License (LGPL) as published
    by the Free Software Foundation; either version 3 of the License, or
    (at your option) any later version.

    As a special exception, the Contributors give you permission to link
    this library with independent modules to produce an executable,
    regardless of the license terms of these independent modules, and to
    copy and distribute the resulting executable under terms of your choice,
    provided that you also meet, for each linked independent module, the
    terms and conditions of the license of that module. An independent
    module is a module which is not derived from or based on this library.
    If you modify this library, you must extend this exception to your
    version of the library.

    libzmq is distributed in the hope that it will be useful, but WITHOUT
    ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
    FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
    License for more details.

    You should have received a copy of the GNU Lesser General Public License
    along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

30
#define __STDC_LIMIT_MACROS // to define SIZE_MAX with older compilers
31
#include "testutil.hpp"
somdoron's avatar
somdoron committed
32

33
void handler (int timer_id, void *arg)
somdoron's avatar
somdoron committed
34
{
35 36
    (void) timer_id; //  Stop 'unused' compiler warnings
    *((bool *) arg) = true;
somdoron's avatar
somdoron committed
37 38
}

39
int sleep_and_execute (void *timers_)
40 41 42 43 44
{
    int timeout = zmq_timers_timeout (timers_);

    //  Sleep methods are inaccurate, so we sleep in a loop until time arrived
    while (timeout > 0) {
45
        msleep (timeout);
46
        timeout = zmq_timers_timeout (timers_);
47 48
    }

49
    return zmq_timers_execute (timers_);
50 51
}

52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
void test_null_timer_pointers ()
{
    void *timers = NULL;

    int rc = zmq_timers_destroy (&timers);
    assert (rc == -1 && errno == EFAULT);

//  TODO this currently triggers an access violation
#if 0
  rc = zmq_timers_destroy (NULL);
  assert (rc == -1 && errno == EFAULT);
#endif

    const size_t dummy_interval = 100;
    const int dummy_timer_id = 1;

    rc = zmq_timers_add (timers, dummy_interval, &handler, NULL);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_add (&timers, dummy_interval, &handler, NULL);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_cancel (timers, dummy_timer_id);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_cancel (&timers, dummy_timer_id);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_set_interval (timers, dummy_timer_id, dummy_interval);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_set_interval (&timers, dummy_timer_id, dummy_interval);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_reset (timers, dummy_timer_id);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_reset (&timers, dummy_timer_id);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_timeout (timers);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_timeout (&timers);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_execute (timers);
    assert (rc == -1 && errno == EFAULT);

    rc = zmq_timers_execute (&timers);
    assert (rc == -1 && errno == EFAULT);
}

105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139
void test_corner_cases ()
{
    void *timers = zmq_timers_new ();
    assert (timers);

    const size_t dummy_interval = SIZE_MAX;
    const int dummy_timer_id = 1;

    //  attempt to cancel non-existent timer
    int rc = zmq_timers_cancel (timers, dummy_timer_id);
    assert (rc == -1 && errno == EINVAL);

    //  attempt to set interval of non-existent timer
    rc = zmq_timers_set_interval (timers, dummy_timer_id, dummy_interval);
    assert (rc == -1 && errno == EINVAL);

    //  attempt to reset non-existent timer
    rc = zmq_timers_reset (timers, dummy_timer_id);
    assert (rc == -1 && errno == EINVAL);

    //  attempt to add NULL handler
    rc = zmq_timers_add (timers, dummy_interval, NULL, NULL);
    assert (rc == -1 && errno == EFAULT);

    int timer_id = zmq_timers_add (timers, dummy_interval, handler, NULL);
    assert (timer_id != -1);

    //  attempt to cancel timer twice
    //  TODO should this case really be an error? canceling twice could be allowed
    rc = zmq_timers_cancel (timers, timer_id);
    assert (rc == 0);

    rc = zmq_timers_cancel (timers, timer_id);
    assert (rc == -1 && errno == EINVAL);

140
    //  timeout without any timers active
141
    rc = zmq_timers_timeout (timers);
142 143
    assert (rc == -1);

144 145 146 147
    rc = zmq_timers_destroy (&timers);
    assert (rc == 0);
}

somdoron's avatar
somdoron committed
148 149 150 151
int main (void)
{
    setup_test_environment ();

152
    void *timers = zmq_timers_new ();
somdoron's avatar
somdoron committed
153 154 155 156
    assert (timers);

    bool timer_invoked = false;

157
    const unsigned long full_timeout = 100;
158 159 160 161
    void *const stopwatch = zmq_stopwatch_start ();

    int timer_id =
      zmq_timers_add (timers, full_timeout, handler, &timer_invoked);
somdoron's avatar
somdoron committed
162 163
    assert (timer_id);

164
    //  Timer should not have been invoked yet
somdoron's avatar
somdoron committed
165 166
    int rc = zmq_timers_execute (timers);
    assert (rc == 0);
167 168 169 170 171 172

#ifdef ZMQ_BUILD_DRAFT_API
    if (zmq_stopwatch_intermediate (stopwatch) < full_timeout) {
        assert (!timer_invoked);
    }
#endif
somdoron's avatar
somdoron committed
173 174

    //  Wait half the time and check again
175
    long timeout = zmq_timers_timeout (timers);
176 177
    assert (rc != -1);
    msleep (timeout / 2);
somdoron's avatar
somdoron committed
178 179
    rc = zmq_timers_execute (timers);
    assert (rc == 0);
180 181 182 183 184
#ifdef ZMQ_BUILD_DRAFT_API
    if (zmq_stopwatch_intermediate (stopwatch) < full_timeout) {
        assert (!timer_invoked);
    }
#endif
somdoron's avatar
somdoron committed
185

186
    // Wait until the end
187
    rc = sleep_and_execute (timers);
somdoron's avatar
somdoron committed
188 189 190 191 192
    assert (rc == 0);
    assert (timer_invoked);
    timer_invoked = false;

    //  Wait half the time and check again
193 194
    timeout = zmq_timers_timeout (timers);
    assert (rc != -1);
195
    msleep (timeout / 2);
somdoron's avatar
somdoron committed
196 197
    rc = zmq_timers_execute (timers);
    assert (rc == 0);
198 199 200 201 202
#ifdef ZMQ_BUILD_DRAFT_API
    if (zmq_stopwatch_intermediate (stopwatch) < 2 * full_timeout) {
        assert (!timer_invoked);
    }
#endif
somdoron's avatar
somdoron committed
203 204 205

    // Reset timer and wait half of the time left
    rc = zmq_timers_reset (timers, timer_id);
206
    assert (rc == 0);
207
    msleep (timeout / 2);
somdoron's avatar
somdoron committed
208 209
    rc = zmq_timers_execute (timers);
    assert (rc == 0);
210 211 212
    if (zmq_stopwatch_stop (stopwatch) < 2 * full_timeout) {
        assert (!timer_invoked);
    }
somdoron's avatar
somdoron committed
213 214

    // Wait until the end
215
    rc = sleep_and_execute (timers);
somdoron's avatar
somdoron committed
216 217 218 219 220
    assert (rc == 0);
    assert (timer_invoked);
    timer_invoked = false;

    // reschedule
221 222
    rc = zmq_timers_set_interval (timers, timer_id, 50);
    assert (rc == 0);
223
    rc = sleep_and_execute (timers);
somdoron's avatar
somdoron committed
224 225 226 227 228 229
    assert (rc == 0);
    assert (timer_invoked);
    timer_invoked = false;

    // cancel timer
    timeout = zmq_timers_timeout (timers);
230 231 232
    assert (rc != -1);
    rc = zmq_timers_cancel (timers, timer_id);
    assert (rc == 0);
233
    msleep (timeout * 2);
somdoron's avatar
somdoron committed
234 235 236 237
    rc = zmq_timers_execute (timers);
    assert (rc == 0);
    assert (!timer_invoked);

238
    rc = zmq_timers_destroy (&timers);
somdoron's avatar
somdoron committed
239 240
    assert (rc == 0);

241
    test_null_timer_pointers ();
242
    test_corner_cases ();
243

somdoron's avatar
somdoron committed
244 245
    return 0;
}