1c67c4c8dSMarouene Boubakri // SPDX-License-Identifier: BSD-2-Clause
2c67c4c8dSMarouene Boubakri /*
3c67c4c8dSMarouene Boubakri * Copyright (c) 2018, Linaro Limited
4c67c4c8dSMarouene Boubakri * Copyright (C) 2017, Fuzhou Rockchip Electronics Co., Ltd.
5c67c4c8dSMarouene Boubakri * All rights reserved.
6c67c4c8dSMarouene Boubakri *
7c67c4c8dSMarouene Boubakri * Redistribution and use in source and binary forms, with or without
8c67c4c8dSMarouene Boubakri * modification, are permitted provided that the following conditions are met:
9c67c4c8dSMarouene Boubakri *
10c67c4c8dSMarouene Boubakri * 1. Redistributions of source code must retain the above copyright notice,
11c67c4c8dSMarouene Boubakri * this list of conditions and the following disclaimer.
12c67c4c8dSMarouene Boubakri *
13c67c4c8dSMarouene Boubakri * 2. Redistributions in binary form must reproduce the above copyright notice,
14c67c4c8dSMarouene Boubakri * this list of conditions and the following disclaimer in the documentation
15c67c4c8dSMarouene Boubakri * and/or other materials provided with the distribution.
16c67c4c8dSMarouene Boubakri *
17c67c4c8dSMarouene Boubakri * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18c67c4c8dSMarouene Boubakri * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19c67c4c8dSMarouene Boubakri * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20c67c4c8dSMarouene Boubakri * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21c67c4c8dSMarouene Boubakri * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22c67c4c8dSMarouene Boubakri * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23c67c4c8dSMarouene Boubakri * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24c67c4c8dSMarouene Boubakri * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25c67c4c8dSMarouene Boubakri * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26c67c4c8dSMarouene Boubakri * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27c67c4c8dSMarouene Boubakri * POSSIBILITY OF SUCH DAMAGE.
28c67c4c8dSMarouene Boubakri */
29c67c4c8dSMarouene Boubakri
30c67c4c8dSMarouene Boubakri #include <kernel/boot.h>
31c67c4c8dSMarouene Boubakri #include <kernel/delay.h>
32c67c4c8dSMarouene Boubakri #include <kernel/misc.h>
33c67c4c8dSMarouene Boubakri
34c67c4c8dSMarouene Boubakri #ifdef CFG_CORE_HAS_GENERIC_TIMER
timeout_elapsed_us(uint64_t expire)35*ccd64a52SEtienne Carriere int timeout_elapsed_us(uint64_t expire)
36*ccd64a52SEtienne Carriere {
37*ccd64a52SEtienne Carriere int64_t diff = delay_cnt_read() - expire;
38*ccd64a52SEtienne Carriere
39*ccd64a52SEtienne Carriere if (MUL_OVERFLOW(diff, 1000000, &diff) ||
40*ccd64a52SEtienne Carriere diff < INT_MIN || diff > INT_MAX) {
41*ccd64a52SEtienne Carriere if (timeout_elapsed(expire))
42*ccd64a52SEtienne Carriere return INT_MAX;
43*ccd64a52SEtienne Carriere else
44*ccd64a52SEtienne Carriere return INT_MIN;
45*ccd64a52SEtienne Carriere }
46*ccd64a52SEtienne Carriere
47*ccd64a52SEtienne Carriere return diff / delay_cnt_freq();
48*ccd64a52SEtienne Carriere }
49*ccd64a52SEtienne Carriere
udelay(uint32_t us)50c67c4c8dSMarouene Boubakri void udelay(uint32_t us)
51c67c4c8dSMarouene Boubakri {
52c67c4c8dSMarouene Boubakri uint64_t target = timeout_init_us(us);
53c67c4c8dSMarouene Boubakri
54c67c4c8dSMarouene Boubakri while (!timeout_elapsed(target))
55c67c4c8dSMarouene Boubakri ;
56c67c4c8dSMarouene Boubakri }
57c67c4c8dSMarouene Boubakri #else
58c67c4c8dSMarouene Boubakri
udelay(uint32_t us)59c67c4c8dSMarouene Boubakri void udelay(uint32_t us)
60c67c4c8dSMarouene Boubakri {
61c67c4c8dSMarouene Boubakri uint64_t cycles = 0;
62c67c4c8dSMarouene Boubakri uint32_t cycles_to_wait = 0;
63c67c4c8dSMarouene Boubakri
64c67c4c8dSMarouene Boubakri cycles = (uint64_t)us * ((uint64_t)plat_get_freq() / 1000000ULL);
65c67c4c8dSMarouene Boubakri
66c67c4c8dSMarouene Boubakri while (cycles) {
67c67c4c8dSMarouene Boubakri cycles_to_wait = MIN(cycles, UINT32_MAX);
68c67c4c8dSMarouene Boubakri wait_cycles(cycles_to_wait);
69c67c4c8dSMarouene Boubakri cycles -= cycles_to_wait;
70c67c4c8dSMarouene Boubakri }
71c67c4c8dSMarouene Boubakri }
72c67c4c8dSMarouene Boubakri #endif
73c67c4c8dSMarouene Boubakri
mdelay(uint32_t ms)74c67c4c8dSMarouene Boubakri void mdelay(uint32_t ms)
75c67c4c8dSMarouene Boubakri {
76c67c4c8dSMarouene Boubakri udelay(1000 * ms);
77c67c4c8dSMarouene Boubakri }
78