mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-11-04 05:17:07 +00:00 
			
		
		
		
	git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@193254 91177308-0d34-0410-b5e6-96231b3b80d8
		
			
				
	
	
		
			149 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*===-- 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 <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <assert.h>
 | 
						|
 | 
						|
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;
 | 
						|
  int i;
 | 
						|
 | 
						|
  for (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 '@': {
 | 
						|
      LLVMValueRef off;
 | 
						|
 | 
						|
      if (depth < 1) {
 | 
						|
        printf("stack underflow\n");
 | 
						|
        return NULL;
 | 
						|
      }
 | 
						|
 | 
						|
      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];
 | 
						|
  LLVMValueRef param;
 | 
						|
  LLVMValueRef res;
 | 
						|
 | 
						|
  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"));
 | 
						|
 | 
						|
  LLVMGetParams(F, ¶m);
 | 
						|
  LLVMSetValueName(param, "in");
 | 
						|
 | 
						|
  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;
 | 
						|
}
 |