Added support for theories
This commit is contained in:
parent
2e94e558c1
commit
c9501805b6
5 changed files with 240 additions and 0 deletions
|
@ -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
|
||||
|
|
79
include/criterion/theories.h
Normal file
79
include/criterion/theories.h
Normal 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_ */
|
|
@ -13,6 +13,7 @@ set(SAMPLES
|
|||
description
|
||||
other-crashes
|
||||
simple
|
||||
theories
|
||||
)
|
||||
|
||||
set(SCRIPTS
|
||||
|
|
19
samples/theories.c
Normal file
19
samples/theories.c
Normal 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
137
src/theories.c
Normal 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);
|
||||
}
|
Loading…
Add table
Reference in a new issue