/*===-- calc.c - tool for testing libLLVM and llvm-c API ------------------===*\ |* *| |* The LLVM Compiler Infrastructure *| |* *| |* This file is distributed under the University of Illinois Open Source *| |* License. See LICENSE.TXT for details. *| |* *| |*===----------------------------------------------------------------------===*| |* *| |* This file implements the --calc command in llvm-c-test. --calc reads lines *| |* from stdin, parses them as a name and an expression in reverse polish *| |* notation and prints a module with a function with the expression. *| |* *| \*===----------------------------------------------------------------------===*/ #include "llvm-c-test.h" #include "llvm-c/Core.h" #include #include #include #include typedef LLVMValueRef (*binop_func_t)(LLVMBuilderRef, LLVMValueRef LHS, LLVMValueRef RHS, const char *Name); static LLVMOpcode op_to_opcode(char op) { switch (op) { case '+': return LLVMAdd; case '-': return LLVMSub; case '*': return LLVMMul; case '/': return LLVMSDiv; case '&': return LLVMAnd; case '|': return LLVMOr; case '^': return LLVMXor; } assert(0 && "unknown operation"); return 0; } #define MAX_DEPTH 32 static LLVMValueRef build_from_tokens(char **tokens, int ntokens, LLVMBuilderRef builder, LLVMValueRef param) { LLVMValueRef stack[MAX_DEPTH]; int depth = 0; for (int i = 0; i < ntokens; i++) { char tok = tokens[i][0]; switch (tok) { case '+': case '-': case '*': case '/': case '&': case '|': case '^': if (depth < 2) { printf("stack underflow\n"); return NULL; } stack[depth - 2] = LLVMBuildBinOp(builder, op_to_opcode(tok), stack[depth - 1], stack[depth - 2], ""); depth--; break; case '@': if (depth < 1) { printf("stack underflow\n"); return NULL; } LLVMValueRef off = LLVMBuildGEP(builder, param, &stack[depth - 1], 1, ""); stack[depth - 1] = LLVMBuildLoad(builder, off, ""); break; default: { char *end; long val = strtol(tokens[i], &end, 0); if (end[0] != '\0') { printf("error parsing number\n"); return NULL; } if (depth >= MAX_DEPTH) { printf("stack overflow\n"); return NULL; } stack[depth++] = LLVMConstInt(LLVMInt64Type(), val, 1); break; } } } if (depth < 1) { printf("stack underflow at return\n"); return NULL; } LLVMBuildRet(builder, stack[depth - 1]); return stack[depth - 1]; } static void handle_line(char **tokens, int ntokens) { char *name = tokens[0]; LLVMModuleRef M = LLVMModuleCreateWithName(name); LLVMTypeRef I64ty = LLVMInt64Type(); LLVMTypeRef I64Ptrty = LLVMPointerType(I64ty, 0); LLVMTypeRef Fty = LLVMFunctionType(I64ty, &I64Ptrty, 1, 0); LLVMValueRef F = LLVMAddFunction(M, name, Fty); LLVMBuilderRef builder = LLVMCreateBuilder(); LLVMPositionBuilderAtEnd(builder, LLVMAppendBasicBlock(F, "entry")); LLVMValueRef param; LLVMGetParams(F, ¶m); LLVMSetValueName(param, "in"); LLVMValueRef res = build_from_tokens(tokens + 1, ntokens - 1, builder, param); if (res) { char *irstr = LLVMPrintModuleToString(M); puts(irstr); LLVMDisposeMessage(irstr); } LLVMDisposeBuilder(builder); LLVMDisposeModule(M); } int calc(void) { tokenize_stdin(handle_line); return 0; }