aboutsummaryrefslogtreecommitdiff
path: root/libsframe/testsuite/libsframe.unwind/tailcall.c
blob: fd3add2d2cfbc9f38fa46572e41b5a5c0f5c1c88 (plain)
1
2
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
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
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
/* Copyright (C) 2022 Free Software Foundation, Inc.

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

   This program 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 General Public License for more details.

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

#include <execinfo.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "sframe-backtrace-api.h"

#ifdef __has_attribute
# if !__has_attribute (noclone)
#  define ATTRIBUTE_NOCLONE
# endif
#endif
#ifndef ATTRIBUTE_NOCLONE
# define ATTRIBUTE_NOCLONE __attribute__((noclone))
#endif

#define BT_BUF_SIZE 16

/* funclist for running tailcall.  */
const char *const func_list[] =
{
  "show_bt",
  "dec",
  "dec",
  "main"
};

void show_bt ()
{
  void *buffer[BT_BUF_SIZE];
  int j, nptrs, err;
  char **strings;

  /* Call the unwinder to get an array of return addresses.  */
  nptrs = sframe_backtrace (buffer, BT_BUF_SIZE, &err);
  if (nptrs == -1 || nptrs != 4)
    {
      printf ("SFrame error: %s (%d)\n", sframe_bt_errmsg (err), nptrs);
      return;
    }

  /* Get these addresses symbolically.  */
  strings = backtrace_symbols (buffer, nptrs);
  if (strings == NULL) {
      printf ("Error in backtrace_symbols");
      return;
  }

  /* Verify the results.  */
  for (j = 0; j < nptrs; j++)
    if (!strstr (strings[j], func_list[j]))
      break;

  free(strings);

  printf ("%s: tailcall test\n", j == nptrs ? "PASS" : "FAIL");
}

/* An example of tail recursive function.  */
void __attribute__((__noinline__)) ATTRIBUTE_NOCLONE
dec (int n)
{
  void *buffer[BT_BUF_SIZE];
  int nptrs, err;

  /* Call the unwinder to get an array of return addresses.  */
  nptrs = sframe_backtrace (buffer, BT_BUF_SIZE, &err);
  if (nptrs == -1)
    {
      printf ("SFrame error: %s (%d)\n", sframe_bt_errmsg (err), nptrs);
      return;
    }

  if (n < 0)
     return;

  if (n == 2)
    show_bt ();

  /* The last executed statement is recursive call.  */
  dec (n-1);
}

int
main (void)
{
  dec (3);
}