/* This testcase is part of GDB, the GNU debugger.
Copyright 1998-2020 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 . */
/* Test GDB's ability to restore saved registers from stack frames
when using the `return' command.
Jim Blandy --- December 1998 */
#include
/* This is the Emacs Lisp expression I used to generate the functions
in this file. If people modify the functions manually, instead of
changing this expression and re-running it, then evaluating this
expression could wipe out their work, so you probably shouldn't
re-run it. But I leave it here for reference.
(defun callee (n) (format "callee%d" n))
(defun caller (n) (format "caller%d" n))
(defun local (n) (format "l%d" n))
(defun local-sum (n)
(if (zerop n) (insert "0")
(let ((j 1))
(while (<= j n)
(insert (local j))
(if (< j n) (insert "+"))
(setq j (1+ j))))))
(defun local-chain (n previous first-end)
(let ((j 1))
(while (<= j n)
(insert " register int " (local j)
" = increment (" previous ");")
(if first-end
(progn
(insert " /" "* " first-end " prologue *" "/")
(setq first-end nil)))
(insert "\n")
(setq previous (local j))
(setq j (1+ j))))
previous)
(save-excursion
(let ((limit 5))
(goto-char (point-max))
(search-backward "generated code starts here")
(forward-line 1)
(let ((start (point)))
(search-forward "generated code ends here")
(forward-line 0)
(delete-region start (point)))
;; Generate callee functions.
(let ((i 0))
(while (<= i limit)
(insert (format "/%s Returns n * %d + %d %s/\n"
"*" i (/ (+ i (* i i)) 2) "*"))
(insert "int\n")
(insert (callee i) " (int n)\n")
(insert "{\n")
(local-chain i "n" (callee i))
(insert " return ")
(local-sum i)
(insert ";\n")
(insert "}\n\n")
(setq i (1+ i))))
;; Generate caller functions.
(let ((i 1))
(while (<= i limit)
(insert "int\n")
(insert (caller i) " (void)\n")
(insert "{\n")
(let ((last (local-chain i "0x7eeb" (caller i))))
(insert " register int n;\n")
(let ((j 0))
(while (<= j limit)
(insert " n = " (callee j) " ("
(if (> j 0) "n + " "")
last ");\n")
(setq j (1+ j)))))
(insert " return n+")
(local-sum i)
(insert ";\n")
(insert "}\n\n")
(setq i (1+ i))))
;; Generate driver function.
(insert "void\n")
(insert "driver (void)\n")
(insert "{\n")
(let ((i 1))
(while (<= i limit)
(insert " printf (\"" (caller i) " () => %d\\n\", "
(caller i) " ());\n")
(setq i (1+ i))))
(insert "}\n\n")))
*/
int
increment (int n)
{
return n + 1;
}
/* generated code starts here */
/* Returns n * 0 + 0 */
int
callee0 (int n)
{
return 0;
}
/* Returns n * 1 + 1 */
int
callee1 (int n)
{
register int l1 = increment (n); /* callee1 prologue */
return l1;
}
/* Returns n * 2 + 3 */
int
callee2 (int n)
{
register int l1 = increment (n); /* callee2 prologue */
register int l2 = increment (l1);
return l1+l2;
}
/* Returns n * 3 + 6 */
int
callee3 (int n)
{
register int l1 = increment (n); /* callee3 prologue */
register int l2 = increment (l1);
register int l3 = increment (l2);
return l1+l2+l3;
}
/* Returns n * 4 + 10 */
int
callee4 (int n)
{
register int l1 = increment (n); /* callee4 prologue */
register int l2 = increment (l1);
register int l3 = increment (l2);
register int l4 = increment (l3);
return l1+l2+l3+l4;
}
/* Returns n * 5 + 15 */
int
callee5 (int n)
{
register int l1 = increment (n); /* callee5 prologue */
register int l2 = increment (l1);
register int l3 = increment (l2);
register int l4 = increment (l3);
register int l5 = increment (l4);
return l1+l2+l3+l4+l5;
}
int
caller1 (void)
{
register int l1 = increment (0x7eeb); /* caller1 prologue */
register int n;
n = callee0 (l1);
n = callee1 (n + l1);
n = callee2 (n + l1);
n = callee3 (n + l1);
n = callee4 (n + l1);
n = callee5 (n + l1);
return n+l1;
}
int
caller2 (void)
{
register int l1 = increment (0x7eeb); /* caller2 prologue */
register int l2 = increment (l1);
register int n;
n = callee0 (l2);
n = callee1 (n + l2);
n = callee2 (n + l2);
n = callee3 (n + l2);
n = callee4 (n + l2);
n = callee5 (n + l2);
return n+l1+l2;
}
int
caller3 (void)
{
register int l1 = increment (0x7eeb); /* caller3 prologue */
register int l2 = increment (l1);
register int l3 = increment (l2);
register int n;
n = callee0 (l3);
n = callee1 (n + l3);
n = callee2 (n + l3);
n = callee3 (n + l3);
n = callee4 (n + l3);
n = callee5 (n + l3);
return n+l1+l2+l3;
}
int
caller4 (void)
{
register int l1 = increment (0x7eeb); /* caller4 prologue */
register int l2 = increment (l1);
register int l3 = increment (l2);
register int l4 = increment (l3);
register int n;
n = callee0 (l4);
n = callee1 (n + l4);
n = callee2 (n + l4);
n = callee3 (n + l4);
n = callee4 (n + l4);
n = callee5 (n + l4);
return n+l1+l2+l3+l4;
}
int
caller5 (void)
{
register int l1 = increment (0x7eeb); /* caller5 prologue */
register int l2 = increment (l1);
register int l3 = increment (l2);
register int l4 = increment (l3);
register int l5 = increment (l4);
register int n;
n = callee0 (l5);
n = callee1 (n + l5);
n = callee2 (n + l5);
n = callee3 (n + l5);
n = callee4 (n + l5);
n = callee5 (n + l5);
return n+l1+l2+l3+l4+l5;
}
void
driver (void)
{
printf ("caller1 () => %d\n", caller1 ());
printf ("caller2 () => %d\n", caller2 ());
printf ("caller3 () => %d\n", caller3 ());
printf ("caller4 () => %d\n", caller4 ());
printf ("caller5 () => %d\n", caller5 ());
}
/* generated code ends here */
int main ()
{
register int local;
driver ();
printf("exiting\n");
return 0;
}