Added support for theories

This commit is contained in:
Snaipe 2015-09-03 06:38:44 +02:00
parent 2e94e558c1
commit c9501805b6
5 changed files with 240 additions and 0 deletions

View file

@ -68,6 +68,7 @@ set(SOURCE_FILES
src/i18n.h
src/ordered-set.c
src/posix-compat.c
src/theories.c
src/main.c
)
@ -91,6 +92,7 @@ set(INTERFACE_FILES
include/criterion/options.h
include/criterion/ordered-set.h
include/criterion/stats.h
include/criterion/theories.h
)
# Generate the configure file
@ -120,6 +122,8 @@ if (COVERALLS)
coveralls_setup("${SOURCE_FILES}" ${COVERALLS_UPLOAD})
endif()
target_link_libraries(criterion dyncall_s)
install(FILES ${INTERFACE_FILES} DESTINATION include/criterion)
install(TARGETS criterion
RUNTIME DESTINATION bin

View file

@ -0,0 +1,79 @@
/*
* The MIT License (MIT)
*
* Copyright © 2015 Franklin "Snaipe" Mathieu <http://snai.pe/>
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
#ifndef CRITERION_THEORIES_H_
# define CRITERION_THEORIES_H_
# include <stddef.h>
# include "criterion.h"
struct criterion_theory_context;
struct criterion_theory_context* cr_theory_init(void);
void cr_theory_push_arg(struct criterion_theory_context *ctx, bool is_float, size_t size, void *ptr);
void cr_theory_free(struct criterion_theory_context *ctx);
void cr_theory_abort(void);
int cr_theory_mark(void);
void cr_theory_reset(struct criterion_theory_context *ctx);
void cr_theory_call(struct criterion_theory_context *ctx, void (*fnptr)(void));
# define TheoryDataPoints(Category, Name) \
static struct criterion_datapoints IDENTIFIER_(Category, Name, dps)[]
# define DataPoints(Type, ...) \
{ sizeof (Type), sizeof ((Type[]) { __VA_ARGS__ }) / sizeof (Type), #Type, &(Type[]) { __VA_ARGS__ } }
struct criterion_datapoints {
size_t size;
size_t len;
const char *name;
void *arr;
};
# define CR_NB_DATAPOINTS(Var) \
(sizeof (Var) / sizeof (struct criterion_datapoints))
# define cr_assume(Condition) \
do { \
if (!(Condition)) \
cr_theory_abort(); \
} while (0);
void cr_theory_main(struct criterion_datapoints *dps, size_t datapoints, void (*fnptr)(void));
# define CR_VAARG_ID(Suffix, Category, Name, ...) \
IDENTIFIER_(Category, Name, Suffix)
# define Theory(Args, ...) \
void CR_VAARG_ID(theory, __VA_ARGS__,)Args; \
Test_(__VA_ARGS__, .sentinel_ = 0) { \
cr_theory_main( \
CR_VAARG_ID(dps, __VA_ARGS__,), \
CR_NB_DATAPOINTS(CR_VAARG_ID(dps, __VA_ARGS__,)), \
(void(*)(void)) CR_VAARG_ID(theory, __VA_ARGS__,) \
); \
} \
void CR_VAARG_ID(theory, __VA_ARGS__,)Args
#endif /* !CRITERION_THEORIES_H_ */

View file

@ -13,6 +13,7 @@ set(SAMPLES
description
other-crashes
simple
theories
)
set(SCRIPTS

19
samples/theories.c Normal file
View file

@ -0,0 +1,19 @@
#include <criterion/theories.h>
TheoryDataPoints(theory, simple) = {
DataPoints(int, 1, 2, 3),
DataPoints(long, 314, 42)
};
Theory((int a, long b), theory, simple) {
cr_assume(a == 2);
printf("%d, %ld\n", a, b);
}
TheoryDataPoints(theory, float) = {
DataPoints(float, 3.14, 42),
};
Theory((float a), theory, float) {
printf("%f\n", (double) a);
}

137
src/theories.c Normal file
View file

@ -0,0 +1,137 @@
/*
* The MIT License (MIT)
*
* Copyright © 2015 Franklin "Snaipe" Mathieu <http://snai.pe/>
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
#include <stdbool.h>
#include <stdarg.h>
#include <setjmp.h>
#include <dyncall.h>
#include <assert.h>
#include "criterion/theories.h"
struct criterion_theory_context {
DCCallVM* vm;
};
void cr_theory_push_arg(struct criterion_theory_context *ctx, bool is_float, size_t size, void *ptr) {
if (is_float) {
if (size == sizeof (float)) {
dcArgFloat(ctx->vm, *(float*)ptr);
} else if (size == sizeof (double)) {
dcArgDouble(ctx->vm, *(double*)ptr);
} else if (size == sizeof (long double)) {
dcArgDouble(ctx->vm, *(long double*)ptr);
}
} else {
if (size == sizeof (char)) {
dcArgChar(ctx->vm, *(char*)ptr);
} else if (size == sizeof (short)) {
dcArgShort(ctx->vm, *(short*)ptr);
} else if (size == sizeof (int)) {
dcArgInt(ctx->vm, *(int*)ptr);
} else if (size == sizeof (bool)) {
dcArgBool(ctx->vm, *(bool*)ptr);
} else if (size == sizeof (long)) {
dcArgLong(ctx->vm, *(long*)ptr);
} else if (size == sizeof (long long)) {
dcArgLongLong(ctx->vm, *(long long*)ptr);
} else if (size == sizeof (void*)) {
dcArgPointer(ctx->vm, *(void**)ptr);
} else {
dcArgPointer(ctx->vm, ptr);
}
}
}
struct criterion_theory_context* cr_theory_init(void) {
struct criterion_theory_context* ctx = malloc(sizeof (struct criterion_theory_context));
ctx->vm = dcNewCallVM(4096);
dcMode(ctx->vm, DC_CALL_C_DEFAULT);
return ctx;
}
void cr_theory_free(struct criterion_theory_context *ctx) {
dcFree(ctx->vm);
}
static jmp_buf theory_jmp;
void cr_theory_abort(void) {
longjmp(theory_jmp, 1);
}
int cr_theory_mark(void) {
return setjmp(theory_jmp);
}
void cr_theory_reset(struct criterion_theory_context *ctx) {
dcReset(ctx->vm);
}
void cr_theory_call(struct criterion_theory_context *ctx, void (*fnptr)(void)) {
dcCallVoid(ctx->vm, (DCpointer) fnptr);
}
static bool contains_word(const char *str, const char *pattern, size_t sz) {
char *res = strstr(str, pattern);
return res
&& (res == str || (res > str && res[-1] == ' '))
&& (!res[sz - 1] || res[sz - 1] == ' ');
}
void cr_theory_main(struct criterion_datapoints *dps, size_t datapoints, void (*fnptr)(void)) {
struct criterion_theory_context *ctx = cr_theory_init();
size_t indices[datapoints];
memset(indices, 0, datapoints * sizeof (size_t));
bool has_next = true;
while (has_next) {
if (!cr_theory_mark()) {
cr_theory_reset(ctx);
for (size_t i = 0; i < datapoints; ++i) {
bool is_float = contains_word(dps[i].name, "float", sizeof ("float"))
|| contains_word(dps[i].name, "double", sizeof ("double"));
cr_theory_push_arg(ctx,
is_float,
dps[i].size,
((char*) dps[i].arr) + dps[i].size * indices[i]);
}
cr_theory_call(ctx, fnptr);
}
for (size_t i = 0; i < datapoints; ++i) {
if (indices[i] == dps[i].len - 1) {
indices[i] = 0;
has_next = i != (datapoints - 1);
} else {
++indices[i];
break;
}
}
}
cr_theory_free(ctx);
}