I'm trying to create a simple dice-rolling application in Pebble using C on CloudPebble. I have an array of different die sizes you can scroll through using Up/Down, and you roll (currently just generate a random number, it'll get fancier later) using the middle button. There's also a label at the top that displays the current die.
It's mostly working, but there's a bug I can't figure out for the life of me:
When it first started, the D6 was broken. I tinkered and tweaked and added some code to see what was going on in the background, but suddenly the D6 started working (no idea why) and the D20 was broken. I tried changing the "20" item in the array to "35" just to see what it would do, and the now-D35 worked, but the D2 and D4 were broken. I changed it back, and different dice broke.
Attached is my current code, which is currently showing D20 and D4 as broken. I've stripped out all the junk and reworking and side code I was using for troubleshooting, since none of it helped.
Can anyone please help explain why the dice are breaking seemingly at random, and how to fix it? Thank you!
#include <pebble.h>
static Window *s_main_window;
static TextLayer *s_label_layer;
static TextLayer *s_output_layer;
static int dice_select = 6;
static char *die_label = "D";
static char *dice_array[] = {"2", "4", "6", "8", "10", "12", "20", "100", NULL};
// === Separate Processes ===
static void update_label(char *die) {
die_label[1] = '\0';
strcat(die_label, die);
text_layer_set_text(s_label_layer, die_label);
}
// === Main Window Processes ===
static void main_window_load(Window *window) {
Layer *window_layer = window_get_root_layer(window);
GRect window_bounds = layer_get_bounds(window_layer);
// Create label TextLayer
s_label_layer = text_layer_create(GRect(5, 0, window_bounds.size.w - 10, 24));
text_layer_set_font(s_label_layer, fonts_get_system_font(FONT_KEY_GOTHIC_18_BOLD));
text_layer_set_text_alignment(s_label_layer, GTextAlignmentCenter);
text_layer_set_text(s_label_layer, "D20");
text_layer_set_overflow_mode(s_label_layer, GTextOverflowModeWordWrap);
layer_add_child(window_layer, text_layer_get_layer(s_label_layer));
// Create output TextLayer
s_output_layer = text_layer_create(GRect(5, 24, window_bounds.size.w - 10, window_bounds.size.h - 24));
text_layer_set_font(s_output_layer, fonts_get_system_font(FONT_KEY_GOTHIC_28));
text_layer_set_text(s_output_layer, "No button pressed yet.");
text_layer_set_overflow_mode(s_output_layer, GTextOverflowModeWordWrap);
layer_add_child(window_layer, text_layer_get_layer(s_output_layer));
}
static void main_window_unload(Window *window) {
// Destroy output TextLayer
text_layer_destroy(s_label_layer);
text_layer_destroy(s_output_layer);
}
// === Button Handlers ===
static void up_click_handler(ClickRecognizerRef recognizer, void *context) {
// Increment dice selection
if(dice_select < 7) {
dice_select += 1;
}
update_label(dice_array[dice_select]);
}
static void select_click_handler(ClickRecognizerRef recognizer, void *context) {
// Establish die selection
char *die = dice_array[dice_select];
int die_sides = atoi(die);
// Roll die
int result = rand() % die_sides + 1;
char *result_str = "";
snprintf(result_str, sizeof(result_str), "%d", result);
// Print result
text_layer_set_text(s_output_layer, result_str);
}
static void down_click_handler(ClickRecognizerRef recognizer, void *context) {
// Increment dice selection
if(dice_select > 0) {
dice_select -= 1;
}
update_label(dice_array[dice_select]);
}
static void click_config_provider(void *context) {
// Register the ClickHandlers
window_single_click_subscribe(BUTTON_ID_UP, up_click_handler);
window_single_click_subscribe(BUTTON_ID_SELECT, select_click_handler);
window_single_click_subscribe(BUTTON_ID_DOWN, down_click_handler);
}
// === Initialize/Deinitialize App ===
static void init() {
// Create main Window
s_main_window = window_create();
window_set_window_handlers(s_main_window, (WindowHandlers) {
.load = main_window_load,
.unload = main_window_unload,
});
window_set_click_config_provider(s_main_window, click_config_provider);
window_stack_push(s_main_window, true);
}
static void deinit() {
// Destroy main Window
window_destroy(s_main_window);
}
int main(void) {
init();
app_event_loop();
deinit();
}
(Note: I'm an untrained self-taught beginner, so the answer to all questions like "Why did you do this that way?" is "Because I didn't know any better.")
EDIT: New code as added per comments.
This part works great:
static char die_label[32] = "D";
static void update_label(char *die) {
snprintf(die_label, sizeof(die_label), "D%s", die);
text_layer_set_text(s_label_layer, die_label);
}
But this part will no longer print the roll result:
// Roll die
int result = rand() % die_sides + 1;
char result_str[32];
snprintf(result_str, sizeof(result_str), "%d", result);
// Print result
text_layer_set_text(s_output_layer, result_str);
The problem is this line
static char *die_label = "D";
That points die_label
to a region of memory that a) should not be written to, and b) only has space for two characters, the D
and the \0
terminator. So the strcat
is writing into memory that it shouldn't be.
The solution is to declare die_label
as
static char die_label[32] = "D"; // reserve space for 32 characters
and then I suggest changing the update function to
static void update_label(char *die)
{
sprintf(die_label, "D%s", die);
text_layer_set_text(s_label_layer, die_label);
}
You have the same problem with these two lines
char *result_str = "";
snprintf(result_str, sizeof(result_str), "%d", result);
In this instance, you only have space for one character, but the length that is passed to snprintf
is the sizeof a pointer, which is either 4 or 8 bytes. I would change those lines to
char result_str[32];
sprintf( result_str, "%d", result );
In response to the updated question:
Yes, I wondered about that. Evidently text_layer_set_text
doesn't make a copy of the string, it just keeps a pointer to it. That means that the result_str
cannot be a local variable, since a local variable will go out of scope as soon as the function returns. The solution is to declare result_str
at the top of the file just like die_label
, or declare it as static
in the function itself.