From: Sascha Hauer <s.hauer@pengutronix.de>
To: Daniel Schultz <d.schultz@phytec.de>
Cc: barebox@lists.infradead.org
Subject: Re: [RFC] commands: Add dhrystone
Date: Mon, 20 Jul 2015 08:47:02 +0200 [thread overview]
Message-ID: <20150720064702.GW18700@pengutronix.de> (raw)
In-Reply-To: <1437054633-14482-1-git-send-email-d.schultz@phytec.de>
Hi Daniel,
On Thu, Jul 16, 2015 at 03:50:33PM +0200, Daniel Schultz wrote:
> This tool will help to measure the system performance.
>
> Signed-off-by: Daniel Schultz <d.schultz@phytec.de>
> ---
> commands/Kconfig | 7 +
> commands/Makefile | 1 +
> commands/dhrystone.c | 404 +++++++++++++++++++++++++++++++++++++++++++++++++++
> 3 files changed, 412 insertions(+)
> create mode 100644 commands/dhrystone.c
>
> diff --git a/commands/Kconfig b/commands/Kconfig
> index bb6674e..9b1109e 100644
> --- a/commands/Kconfig
> +++ b/commands/Kconfig
> @@ -2102,6 +2102,13 @@ config CMD_STATE
> depends on STATE
> prompt "state"
>
> +config CMD_DHRYSTONE
> + bool
> + default n
'n' already is the default. Please drop this line.
> + prompt "dhrystone"
> + help
> + CPU benchmark tool
> +
> # end Miscellaneous commands
> endmenu
>
> diff --git a/commands/Makefile b/commands/Makefile
> index 3698347..879caec 100644
> --- a/commands/Makefile
> +++ b/commands/Makefile
> @@ -112,3 +112,4 @@ obj-$(CONFIG_CMD_NV) += nv.o
> obj-$(CONFIG_CMD_DEFAULTENV) += defaultenv.o
> obj-$(CONFIG_CMD_STATE) += state.o
> obj-$(CONFIG_CMD_DHCP) += dhcp.o
> +obj-$(CONFIG_CMD_DHRYSTONE) += dhrystone.o
> diff --git a/commands/dhrystone.c b/commands/dhrystone.c
> new file mode 100644
> index 0000000..86b65d1
> --- /dev/null
> +++ b/commands/dhrystone.c
> @@ -0,0 +1,404 @@
> +/*
> + * (C) Copyright 2014 Phytec Messtechnik GmbH
> + * Author: Stefan Müller-Klieser <s.mueller-klieser@phytec.de>
> + * (C) Copyright (C) 2015 Phytec Messtechnik GmbH
> + * Author: Daniel Schultz <d.schultz@phytec.de>
> + *
> + * based on "DHRYSTONE" Benchmark Program
> + * Version: C, Version 2.1
> + * Date: May 25, 1988
> + * Author: Reinhold P. Weicker
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2
> + * as published by the Free Software Foundation.
> + *
> + * 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.
> + *
> + */
> +
> +#include <common.h>
> +#include <command.h>
> +#include <errno.h>
> +#include <clock.h>
> +#include <asm-generic/div64.h>
> +#include <malloc.h>
> +#include <stdbool.h>
> +#include <stdio.h> /* for strcpy, strcmp */
> +
> +enum idents {ident_1, ident_2, ident_3, ident_4, ident_5};
> +
> +/* General definitions: */
> +
> +struct record {
> + struct record *ptr_comp;
> + enum idents discr;
> + union {
> + struct {
> + enum idents enum_comp;
> + int int_comp;
> + char str_comp[31];
> + } var_1;
> + struct {
> + enum idents enum_comp_2;
> + char str_2_comp[31];
> + } var_2;
> + struct {
> + char char_1_Comp;
> + char char_2_Comp;
> + } var_3;
> + } variant;
> +};
> +
> +/* Global Variables: */
> +
> +struct record *record_glob;
> +struct record *next_record_glob;
> +int int_glob;
> +bool bool_glob;
> +char char_1_glob;
> +char char_2_glob;
> +int arr_1_glob[50];
> +int arr_2_glob[50][50];
> +
> +/* variables for time measurement: */
> +
> +#define TOO_SMALL_TIME 50000000
50 * MSECOND
> +/* Measurements should last at least 50mseconds */
> +#define TOO_LARGE_TIME 2000000000
2 * SECOND
Makes this better readable.
> +
> +u64 begin_time;
> +u64 end_time;
> +u64 user_time;
> +u64 microseconds;
> +u64 dhrystones_per_second;
> +/* end of variables for time measurement */
> +
> +enum idents compare_chars(char char_1, char char_2)
> +{
> + if (char_1 != char_2) {
> + return ident_1;
> + } else { /* should not executed */
> + char_1_glob = char_1;
> + return ident_2;
> + }
> +}
> +
> +bool compare_strs(char str_1[31], char str_2[31])
> +{
> + int offset;
> +
> + offset = 2;
> + while (offset <= 2)
> + if (compare_chars(str_1[offset], str_2[offset+1]) == ident_1)
> + ++offset;
> + if (strcmp(str_1, str_2) > 0) {
> + int_glob = offset + 7;
> + return true;
> + } else {
> + return false;
> + }
> +
> +}
> +
> +bool check_ident(enum idents ident)
> +{
> + if (ident == ident_3)
> + return true;
> + else
> + return false;
> +}
> +
> +void proc_7(int input_1, int input_2, int *out)
> +{
> + *out = input_2 + input_1 + 2;
> +}
> +
> +void proc_6(enum idents ident, enum idents *ident_out)
> +{
> + *ident_out = ident;
> + if (!check_ident(ident))
> + *ident_out = ident_4;
> + switch (ident) {
> + case ident_1:
> + *ident_out = ident_1;
> + break;
> + case ident_2:
> + if (int_glob > 100)
> + *ident_out = ident_1;
> + else
> + *ident_out = ident_4;
> + break;
> + case ident_3:
> + *ident_out = ident_2;
> + break;
> + case ident_4:
> + break;
> + case ident_5:
> + *ident_out = ident_3;
> + break;
> + }
> +}
> +
> +void fill_record(struct record *record)
> +{
> + struct record *next_record = record->ptr_comp;
> +
> + *record->ptr_comp = *record_glob;
> + record->variant.var_1.int_comp = 5;
> + next_record->variant.var_1.int_comp = record->variant.var_1.int_comp;
> + next_record->ptr_comp = record->ptr_comp;
> + proc_7(10, int_glob, &record_glob->variant.var_1.int_comp);
> + /* Ptr_Val_Par->ptr_comp->ptr_comp == record_glob->ptr_comp */
> + if (next_record->discr == ident_1) { /* then, executed */
> + next_record->variant.var_1.int_comp = 6;
> + proc_6(record->variant.var_1.enum_comp,
> + &next_record->variant.var_1.enum_comp);
> + next_record->ptr_comp = record_glob->ptr_comp;
> + proc_7(next_record->variant.var_1.int_comp, 10,
> + &next_record->variant.var_1.int_comp);
> + } else { /* not executed */
> + *record = *record->ptr_comp;
> + }
> +}
> +
> +void proc_2(int *out)
> +{
> + if (char_1_glob == 'A')
> + *out = *out + 9 - int_glob;
> +}
> +
> +void proc_4(void)
> +{
> + bool_glob = (char_1_glob == 'A') | bool_glob;
> + char_2_glob = 'B';
> +}
> +
> +void proc_5(void)
> +{
> + char_1_glob = 'A';
> + bool_glob = false;
> +}
> +
> +/* dhry_2.c */
> +
> +void fill_array(int arr_1[50], int arr_2[50][50], int val_1, int val_2)
> +{
> + int i;
> + int offset;
> +
> + offset = val_1 + 5;
> + arr_1[offset] = val_2;
> + arr_1[offset+1] = arr_1[offset];
> + arr_1[offset+30] = offset;
> + for (i = offset; i <= offset+1; ++i)
> + arr_2[offset][i] = offset;
> + arr_2[offset][offset-1] += 1;
> + arr_2[offset+20][offset] = arr_1[offset];
> + int_glob = 5;
> +}
> +
> +static int do_dhrystone(int argc, char *argv[])
> +{
> + /* main program, corresponds to procedures */
> + /* Main and Proc_0 in the Ada version */
> + int int_1;
> + int int_2;
> + int int_3;
> + char char_i;
> + enum idents ident;
> + char str_1[31];
> + char str_2[31];
> + int i;
> + int number_of_runs;
> +
> + /* barebox cmd */
> + if (argc < 2)
> + return COMMAND_ERROR_USAGE;
> + number_of_runs = simple_strtoul(argv[1], NULL, 10);
> +
> + /* Initializations */
> + ident = ident_2; /* prevent compiler warning */
> + int_2 = 0; /* prevent compiler warning */
> + int_3 = 0; /* prevent compiler warning */
> +
> + next_record_glob = malloc(sizeof(struct record));
> + record_glob = malloc(sizeof(struct record));
> +
> + record_glob->ptr_comp = next_record_glob;
> + record_glob->discr = ident_1;
> + record_glob->variant.var_1.enum_comp = ident_3;
> + record_glob->variant.var_1.int_comp = 40;
> + strcpy(record_glob->variant.var_1.str_comp,
> + "DHRYSTONE PROGRAM, SOME STRING");
> + strcpy(str_1, "DHRYSTONE PROGRAM, 1'ST STRING");
> +
> + arr_2_glob[8][7] = 10;
> + /* Was missing in published program. Without this statement, */
> + /* arr_2_glob [8][7] would have an undefined value. */
> + /* Warning: With 16-Bit processors and number_of_runs > 32000, */
> + /* overflow may occur for this array element. */
> +
> + printf("\n");
> + printf("Dhrystone Benchmark, Version 2.1 (Language: C)\n");
> + printf("\n");
> + printf("Program compiled without 'register' attribute\n");
> + printf("\n");
> + printf("Execution starts, %d runs through Dhrystone\n", number_of_runs);
> +
> + /***************/
> + /* Start timer */
> + /***************/
> + begin_time = get_time_ns();
> +
> + for (i = 1; i <= number_of_runs; ++i) {
> + proc_5();
> + proc_4();
> + /* char_1_glob == 'A', char_2_glob == 'B', bool_glob == true */
> + int_1 = 2;
> + int_2 = 3;
> + strcpy(str_2, "DHRYSTONE PROGRAM, 2'ND STRING");
> + ident = ident_2;
> + bool_glob = !compare_strs(str_1, str_2);
> + /* bool_glob == 1 */
> + while (int_1 < int_2) {
> + int_3 = 5 * int_1 - int_2;
> + /* int_3 == 7 */
> + proc_7(int_1, int_2, &int_3);
> + /* int_3 == 7 */
> + int_1 += 1;
> + }
> + /* int_1 == 3, int_2 == 3, int_3 == 7 */
> + fill_array(arr_1_glob, arr_2_glob, int_1, int_3);
> + /* int_glob == 5 */
> + fill_record(record_glob);
> + for (char_i = 'A'; char_i <= char_2_glob; ++char_i) {
> + if (ident == compare_chars(char_i, 'C')) {
> + proc_6(ident_1, &ident);
> + strcpy(str_2,
> + "DHRYSTONE PROGRAM, 3'RD STRING");
> + int_2 = i;
> + int_glob = i;
> + }
> + }
> + /* int_1 == 3, int_2 == 3, int_3 == 7 */
> + int_2 = int_2 * int_1;
> + int_1 = int_2 / int_3;
> + int_2 = 7 * (int_2 - int_3) - int_1;
> + /* int_1 == 1, int_2 == 13, int_3 == 7 */
> + proc_2(&int_1);
> + /* int_1 == 5 */
> + }
> +
> + /**************/
> + /* Stop timer */
> + /**************/
> + end_time = get_time_ns();
> +
> + printf("Execution ends\n");
> + printf("\n");
> + printf("Final values of the variables used in the benchmark:\n");
> + printf("\n");
> + printf("int_glob: %d\n", int_glob);
> + printf(" should be: %d\n", 5);
> + printf("bool_glob: %d\n", bool_glob);
> + printf(" should be: %d\n", 1);
> + printf("char_1_glob: %c\n", char_1_glob);
> + printf(" should be: %c\n", 'A');
> + printf("char_2_glob: %c\n", char_2_glob);
> + printf(" should be: %c\n", 'B');
> + printf("arr_1_glob[8]: %d\n", arr_1_glob[8]);
> + printf(" should be: %d\n", 7);
> + printf("arr_2_glob[8][7]: %d\n", arr_2_glob[8][7]);
> + printf(" should be: number_of_runs + 10\n");
> + printf("record_glob->\n");
> + printf(" ptr_comp: %d\n", (int) record_glob->ptr_comp);
> + printf(" should be: (implementation-dependent)\n");
> + printf(" discr: %d\n", record_glob->discr);
> + printf(" should be: %d\n", 0);
> + printf(" enum_comp: %d\n",
> + record_glob->variant.var_1.enum_comp);
> + printf(" should be: %d\n", 2);
> + printf(" int_comp: %d\n",
> + record_glob->variant.var_1.int_comp);
> + printf(" should be: %d\n", 17);
> + printf(" str_comp: %s\n",
> + record_glob->variant.var_1.str_comp);
> + printf(" should be: DHRYSTONE PROGRAM, SOME STRING\n");
> + printf("next_record_glob->\n");
> + printf(" ptr_comp: %d\n", (int) next_record_glob->ptr_comp);
> + printf(" should be: (implementation-dependent), same as above\n");
> + printf(" discr: %d\n", next_record_glob->discr);
> + printf(" should be: %d\n", 0);
> + printf(" enum_comp: %d\n",
> + next_record_glob->variant.var_1.enum_comp);
> + printf(" should be: %d\n", 1);
> + printf(" int_comp: %d\n",
> + next_record_glob->variant.var_1.int_comp);
> + printf(" should be: %d\n", 18);
> + printf(" str_comp: %s\n",
> + next_record_glob->variant.var_1.str_comp);
> + printf(" should be: DHRYSTONE PROGRAM, SOME STRING\n");
> + printf("int_1: %d\n", int_1);
> + printf(" should be: %d\n", 5);
> + printf("int_2: %d\n", int_2);
> + printf(" should be: %d\n", 13);
> + printf("int_3: %d\n", int_3);
> + printf(" should be: %d\n", 7);
> + printf("ident: %d\n", ident);
> + printf(" should be: %d\n", 1);
> + printf("str_1: %s\n", str_1);
> + printf(" should be: DHRYSTONE PROGRAM, 1'ST STRING\n");
> + printf("str_2: %s\n", str_2);
> + printf(" should be: DHRYSTONE PROGRAM, 2'ND STRING\n");
> + printf("\n");
When the values above match the expected values they seem mostly
uninteresting. You should only print them when they actually don't match
the expected values.
> +
> + user_time = end_time - begin_time;
> +
> + if (user_time < TOO_SMALL_TIME) {
> + printf("Measured time too small to obtain meaningful results\n");
> + printf("or a timer wrap happend. Please increase number of runs\n");
> + printf("starting from 10 in *10 increments\n");
> + printf("user_time: %llu\n", user_time);
Add unit to printf (ns)?
> + } else if (user_time > TOO_LARGE_TIME) {
> + printf("Measured time too large to obtain meaningful results.\n");
> + printf("Please decrease number of runs by *10\n");
> + printf("Timer overflow could occured.\n");
> + printf("user_time: %llu\n", user_time);
ditto.
The above loks very 80ies style. How about making the number of
iterations optional on the command line and just run the benchmark until
enough time is over? You may test for the time only every 10000
iterations or so to keep the overhead of measuring the time small.
> + } else if (end_time < begin_time) {
> + printf("Timer overflow occured.\n");
> + printf("Please restart the programm.\n");
> + } else {
> + printf("user_time: %llu ns\n", user_time);
> +
> + microseconds = user_time;
> + do_div(microseconds, number_of_runs);
> +
> + printf("Nanoseconds for one run through Dhrystone: ");
> + printf("%llu\n", microseconds);
Should the variable name be nanoseconds?
Should be a single printf.
> + printf("Dhrystones per Second:\n ");
> + printf("(%d / %llu) * 10^9\n", number_of_runs, user_time);
> + printf("DMIPS:\n ");
> + printf("((%d / %llu) * 10^9) / 1757\n", number_of_runs,
> + user_time);
> + printf("\n");
> + }
> +
> + return 0;
> +}
> +
> +BAREBOX_CMD_HELP_START(dhrystone)
> +BAREBOX_CMD_HELP_TEXT(
> +"this command runs a dhrystone benchmark to get an astimation of the CPU freq")
s/astimation/estimation/
Sascha
--
Pengutronix e.K. | |
Industrial Linux Solutions | http://www.pengutronix.de/ |
Peiner Str. 6-8, 31137 Hildesheim, Germany | Phone: +49-5121-206917-0 |
Amtsgericht Hildesheim, HRA 2686 | Fax: +49-5121-206917-5555 |
_______________________________________________
barebox mailing list
barebox@lists.infradead.org
http://lists.infradead.org/mailman/listinfo/barebox
prev parent reply other threads:[~2015-07-20 6:47 UTC|newest]
Thread overview: 3+ messages / expand[flat|nested] mbox.gz Atom feed top
2015-07-16 13:50 Daniel Schultz
2015-07-16 16:02 ` Antony Pavlov
2015-07-20 6:47 ` Sascha Hauer [this message]
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20150720064702.GW18700@pengutronix.de \
--to=s.hauer@pengutronix.de \
--cc=barebox@lists.infradead.org \
--cc=d.schultz@phytec.de \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox