X-Git-Url: https://git.m6w6.name/?a=blobdiff_plain;f=src%2Ftypes%2Fassert_stmt.c;fp=src%2Ftypes%2Fassert_stmt.c;h=084f9fc192431803b12d1c99d8875bf27fe4b93f;hb=f15288e185e65bcdfde87c217fb6bd7ff12eb653;hp=0000000000000000000000000000000000000000;hpb=e11280a7b6ed67a4d79b14569ca06131ea2338bd;p=m6w6%2Fext-psi diff --git a/src/types/assert_stmt.c b/src/types/assert_stmt.c new file mode 100644 index 0000000..084f9fc --- /dev/null +++ b/src/types/assert_stmt.c @@ -0,0 +1,102 @@ +/******************************************************************************* + Copyright (c) 2017, Michael Wallner . + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE + FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER + CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, + OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*******************************************************************************/ + +#include "php_psi_stdinc.h" + +#include "data.h" +#include "calc.h" + +struct psi_assert_stmt *psi_assert_stmt_init(enum psi_assert_kind kind, struct psi_num_exp *exp) +{ + struct psi_assert_stmt *stmt = calloc(1, sizeof(*stmt)); + + stmt->kind = kind; + stmt->exp = exp; + + return stmt; +} + +void psi_assert_stmt_free(struct psi_assert_stmt **stmt_ptr) +{ + if (*stmt_ptr) { + struct psi_assert_stmt *stmt = *stmt_ptr; + + psi_num_exp_free(&stmt->exp); + if (stmt->token) { + free(stmt->token); + } + free(stmt); + *stmt_ptr = NULL; + } +} + +void psi_assert_stmt_dump(int fd, struct psi_assert_stmt *stmt) +{ + dprintf(fd, "\t%s_assert ", stmt->kind == PSI_ASSERT_PRE ? "pre" : "post"); + psi_num_exp_dump(fd, stmt->exp); + dprintf(fd, ";\n"); +} + +bool psi_assert_stmt_exec(struct psi_assert_stmt *stmt, struct psi_call_frame *frame) +{ + impl_val res, chk; + token_t res_type = psi_num_exp_exec(stmt->exp, &res, frame); + + psi_calc_cast(res_type, &res, PSI_T_UINT8, &chk); + return chk.u8; +} + +bool psi_assert_stmts_validate(struct psi_data *data, struct psi_impl *impl) +{ + size_t i = 0; + struct psi_assert_stmt *ass; + + /* we can have multiple assert stmts */ + while (psi_plist_get(impl->stmts.ass, i++, &ass)) { + if (!psi_num_exp_validate(data, ass->exp, impl, NULL, NULL, NULL, NULL)) { + return false; + } + } + + return true; +} + +char *psi_assert_stmt_message(struct psi_assert_stmt *stmt) +{ + /* FIXME */ + struct stat sb; + char *message, template[] = "psi.assert.XXXXXX"; + int fd = mkstemp(template); + + dprintf(fd, "Failed asserting that "); + psi_num_exp_dump(fd, stmt->exp); + fstat(fd, &sb); + message = malloc(sb.st_size + 1); + lseek(fd, 0, SEEK_SET); + read(fd, message, sb.st_size); + close(fd); + message[sb.st_size] = '\0'; + return message; +}