Basic脚本解释器移植到STM32

本文来自http://blog.csdn.net/hellogv/ 。引用必须注明出处!

上次讲了LUA移植到STM32。这次讲讲Basic脚本解释器移植到STM32。

在STM32上跑Basic脚本,相同能够跟穿戴设备结合。也能够作为刚開始学习的人学习MCU的入门工具,当然前提是有人做好Basic的STM32交互实现。这里使用的是uBasic开源脚本解释器(http://dunkels.com/adam/ubasic/),只是uBasic不支持完整的Basic算法,所以用起来略费心,假设有好的Basic开源脚本解释器。ANSI-C实现的,欢迎推荐。。

本文实现的功能是输入下面basic脚本:

10 v=1
20 for p = 4 to 7
40 write "gpioa",p,v
50 next p
60 if v=0 then goto 10
70 if v=1 then v=0
80 goto 20
run

实现的功能是同一时候把4个LED灯同一时候开后再同一时候关,通过自己定义的命令 write来实现,p是IO脚。v是IO的数值。

write "gpioa",p,v

例如以下图:

本文代码能够到这里下载http://download.csdn.net/detail/hellogv/7391265

main.c的源代码例如以下,通过USART1来发送Basic脚本到STM32,另外还要通过readline()来做些预处理,比如收到“run”这个字符串就表示脚本结束開始执行:

#include "stm32f10x_lib.h"
#include <assert.h>
#include "stdio.h"
#include <stdlib.h>
#include <string.h>
#include "ubasic.h"

/*******************************************************************************
 * 函数名  : RCC_Configuration
 * 函数描写叙述  : 设置系统各部分时钟
 *******************************************************************************/

void RCC_Configuration(void) {
  /* 定义枚举类型变量 HSEStartUpStatus */
  ErrorStatus HSEStartUpStatus;

  /* 复位系统时钟设置*/
  RCC_DeInit();
  /* 开启HSE*/
  RCC_HSEConfig(RCC_HSE_ON );
  /* 等待HSE起振并稳定*/
  HSEStartUpStatus = RCC_WaitForHSEStartUp();
  /* 推断HSE起是否振成功,是则进入if()内部 */
  if (HSEStartUpStatus == SUCCESS) {
    /* 选择HCLK(AHB)时钟源为SYSCLK 1分频 */
    RCC_HCLKConfig(RCC_SYSCLK_Div1 );
    /* 选择PCLK2时钟源为 HCLK(AHB) 1分频 */
    RCC_PCLK2Config(RCC_HCLK_Div1 );
    /* 选择PCLK1时钟源为 HCLK(AHB) 2分频 */
    RCC_PCLK1Config(RCC_HCLK_Div2 );
    /* 设置FLASH延时周期数为2 */
    FLASH_SetLatency(FLASH_Latency_2 );
    /* 使能FLASH预取缓存 */
    FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable );
    /* 选择锁相环(PLL)时钟源为HSE 1分频。倍频数为9。则PLL输出频率为 8MHz * 9 = 72MHz */
    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9 );
    /* 使能PLL */
    RCC_PLLCmd(ENABLE);
    /* 等待PLL输出稳定 */
    while (RCC_GetFlagStatus(RCC_FLAG_PLLRDY ) == RESET)
      ;
    /* 选择SYSCLK时钟源为PLL */
    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK );
    /* 等待PLL成为SYSCLK时钟源 */
    while (RCC_GetSYSCLKSource() != 0x08)
      ;
  }

  /* 开启USART1和GPIOA时钟 */
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA,
      ENABLE);
}

/*******************************************************************************
 * 函数名      : GPIO_Configuration
 * 函数描写叙述      : 设置各GPIOport功能
 *******************************************************************************/

void GPIO_Configuration(void) {
  /* 定义GPIO初始化结构体 GPIO_InitStructure */
  GPIO_InitTypeDef GPIO_InitStructure;

  /* 设置USART1的Tx脚(PA.9)为第二功能推挽输出功能 */
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_Init(GPIOA, &GPIO_InitStructure);

  /* 设置USART1的Rx脚(PA.10)为浮空输入脚 */
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  GPIO_Init(GPIOA, &GPIO_InitStructure);
}

/*******************************************************************************
 * 函数名      : USART_Configuration
 * 函数描写叙述      : 设置USART1
 *******************************************************************************/

void USART_Configuration(void) {
  /* 定义USART初始化结构体 USART_InitStructure */
  USART_InitTypeDef USART_InitStructure;
  /* 定义USART初始化结构体 USART_ClockInitStructure */
  USART_ClockInitTypeDef USART_ClockInitStructure;

  /*  波特率为115200bps;
   *  8位数据长度;
   *  1个停止位。无校验;
   *  禁用硬件流控制;
   *  禁止USART时钟;
   *  时钟极性低;
   *  在第2个边沿捕获数据
   *  最后一位数据的时钟脉冲不从 SCLK 输出;
   */

  USART_ClockInitStructure.USART_Clock = USART_Clock_Disable;
  USART_ClockInitStructure.USART_CPOL = USART_CPOL_Low;
  USART_ClockInitStructure.USART_CPHA = USART_CPHA_2Edge;
  USART_ClockInitStructure.USART_LastBit = USART_LastBit_Disable;
  USART_ClockInit(USART1, &USART_ClockInitStructure);

  USART_InitStructure.USART_BaudRate = 9600;
  USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  USART_InitStructure.USART_StopBits = USART_StopBits_1;
  USART_InitStructure.USART_Parity = USART_Parity_No ;
  USART_InitStructure.USART_HardwareFlowControl =USART_HardwareFlowControl_None;
  USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  USART_Init(USART1, &USART_InitStructure);

  /* 使能USART1 */
  USART_Cmd(USART1, ENABLE);
}

int fputc(int ch, FILE *f) {
  USART_SendData(USART1, (u8) ch);
  while (USART_GetFlagStatus(USART1, USART_FLAG_TC ) == RESET);
  return ch;
}

int getKey(void) {
  while (!(USART1 ->SR & USART_FLAG_RXNE ));
  return ((int) (USART1 ->DR & 0x1FF));
}

/*******************************************************************************
 * 函数名      : readLine
 * 函数描写叙述      : 从串口读取Basic代码
 *******************************************************************************/
bool readLines(char *s) {
  bool isString = FALSE; //推断是否字符串
  char ch;
  char *p = s;

  if(*p!=‘\0‘)
    return FALSE;

  while (1) {
    ch = getKey();
    if (ch == ‘\"‘) { //检測到字符串
      isString = !isString;
    }

    if (ch == ‘\r‘) //屏蔽‘\r‘这个字符
      continue;
    else {
      if (isString) //不改变代码中字符串的大写和小写
        *p++ = ch;
      else //关键字都转为小写
        *p++ = tolower(ch);
    }

    if (*(p-3) == ‘r‘
          &&*(p-2)==‘u‘
          &&*(p-1)==‘n‘){ //run表示程序结束
      *(p-3) = ‘\0‘;
      break;
    }
  }
  return TRUE;
}

#define _MAX_LINE_LENGTH 256
int main(void) {
  /* 设置系统时钟 */
  RCC_Configuration();
  /* 设置GPIOport */
  GPIO_Configuration();
  /* 设置USART */
  USART_Configuration();

  char line[_MAX_LINE_LENGTH];
  while(1){
    memset(line, 0, _MAX_LINE_LENGTH);
    if(readLines(line)){
      ubasic_init(line);
      do {
        ubasic_run();
      } while(!ubasic_finished());
    }
  };

  return 0;

}

实现write命令的源代码例如以下:

#include "tokenizer.h"
#include "stm32f10x_lib.h"
#include <string.h>
#include <ctype.h>
#include <stdlib.h>
GPIO_TypeDef *gpio_x;
u16 gpio_pin_x;
u16 gpio_value;
struct GPIO_KEYWORD gpio_kt;
struct PIN_KEYWORD pin_kt;

#define GET_ARRAY_LEN(array,len){len = (sizeof(array) / sizeof(array[0]));}

struct GPIO_KEYWORD {
  char *keyword;
  GPIO_TypeDef *token;
};

static const struct GPIO_KEYWORD gpio_keywords[] =
{ { "GPIOA", GPIOA },
    { "GPIOB", GPIOB  },
    { "GPIOC", GPIOC  },
    { "GPIOD", GPIOD  }};

struct PIN_KEYWORD {
  u16 keyword;
  u16 token;;
};

static const struct PIN_KEYWORD pin_keywords[17] =
{  { 0, GPIO_Pin_0 },
    { 1, GPIO_Pin_1 },
    { 2, GPIO_Pin_2 },
    { 3, GPIO_Pin_3 },
    { 4, GPIO_Pin_4 },
    { 5, GPIO_Pin_5 },
    { 6, GPIO_Pin_6 },
    { 7, GPIO_Pin_7 },
    { 8, GPIO_Pin_8 },
    { 9, GPIO_Pin_9 },
    { 10, GPIO_Pin_10 },
    { 11, GPIO_Pin_11 },
    { 12, GPIO_Pin_12 },
    { 13, GPIO_Pin_13 },
    { 14, GPIO_Pin_14 },
    { 15, GPIO_Pin_15 }};

void init_my_statement(){
  gpio_x=NULL;
  gpio_pin_x=NULL;
  gpio_value=NULL;
}

void put_value(u16 value){
  gpio_value=value;
}

u16 get_value(){
  return gpio_value;
}

/**
**获取GPIO_PIN_X
**/
bool put_pin(u16 pin){
    int size=0;

    if(gpio_pin_x==NULL){
      GET_ARRAY_LEN(pin_keywords,size);
      if(pin<size){
        gpio_pin_x = pin_keywords[pin].token;
        printf ("------P");printf ("%d\r\n",pin);
        return TRUE;
      }
    }
    return FALSE;
}

u16 get_pin(){
  return gpio_pin_x;
}

/**
**获取获取GPIO_X
**/
bool put_gpio(char* p){
    int i=0;
    int size=0;

    if(gpio_x==NULL){
      GET_ARRAY_LEN(gpio_keywords,size);
      for (i=0; i<size; i++) {
        gpio_kt = gpio_keywords[i];
        if (strncasecmp(p, gpio_kt.keyword, strlen(p)) == 0) {
          printf ("------");printf ("%s\r\n",gpio_kt.keyword);
          gpio_x = gpio_kt.token;
          return TRUE;
        }
      }
    }
    return FALSE;
}

GPIO_TypeDef * get_gpio(){
  return gpio_x;
}

bool write_gpio(){

  //USART1不能被写
  if(gpio_x== GPIOA
     && (gpio_pin_x==GPIO_Pin_9
         || gpio_pin_x==GPIO_Pin_10)){
    return FALSE;
  }

  GPIO_InitTypeDef GPIO_InitStructure;

  GPIO_InitStructure.GPIO_Pin = gpio_pin_x;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  GPIO_Init(gpio_x , &GPIO_InitStructure); 

  GPIO_WriteBit(gpio_x , gpio_pin_x,(BitAction)gpio_value);
  return TRUE;
}

时间: 2024-11-22 16:58:22

Basic脚本解释器移植到STM32的相关文章

单片机脚本语言-移植lua到stm32-MDK

Lua简介 Lua[1]  是一个小巧的脚本语言.作者是巴西人.该语言的设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能. Lua脚本可以很容易的被C/C++ 代码调用,也可以反过来调用C/C++的函数,这使得Lua在应用程序中可以被广泛应用.不仅仅作为扩展脚本,也可以作为普通的配置文件,代替XML,Ini等文件格式,并且更容易理解和维护. Lua的目标是成为一个很容易嵌入其它语言中使用的语言.大多数程序员也认为它的确做到了这一点. 很多应用程序使用LUA作为自己的嵌入式脚本

Ch/CINT — C/C++语言脚本解释器

1.CH Ch是一个跨平台的C/C++脚本解释器,它支持ISO的C语言标准(C90/C99),C++,附带了8000多个函数库并支持众多的工业标准,支持POSIX, socket/Winsock, X11/Motif, OpenGL, ODBC, C LAPACK, GTK+, Win32, XML, 和CGI等等.Ch具有MATLAB的高级数值计算和绘图的功能,且具有良好的交互性,在C/C++语言编程学习方面尤为好用(试试就知道了).Ch也是一个可嵌入的脚本引擎,可以无缝地嵌入到自己的程序中.

将GY-302感光模块从51移植到stm32

这个感光模块主要就是用i2c通信,移植的时候其实就是改一下延时,把端口的模式设置正确就行了 为了尽量减少工作量,我尽量多用宏定义定义,少修改代码,对于延时函数,可以用define 把它替换成stm32里的延时函数 对于变量,注意在stm32里int是四个字节,而51是两个字节,再者就是很关键的一点,51里有bit变量,可以位寻址  而stm32却没有,而且也没有sbit,对于前者,我就直接把它变成uchar,而后者是有解决办法的,就是位带操作. 位带操作就是,设定了一块特殊的地方,映射到其他地址

实现脚本解释器 - 词法分析器

本系列介绍 笔者最近正学习编译原理,为了将理论变为实践,所以创作了本系列来记录学习过程中的思考与问题,注意文章中为了理论上描述方便增加了自创的术语. 本系列使用 Java 语言来实现一个脚本解释器,该脚本语言命名为 Foo,其语法参考 JavaScript 语言,本系列代码地址 Github . 词法分析器介绍 词法分析器的作用是将输入的字符串转变为一个个的记号(token),记号是由记号名(name)和属性值(value)构成的二元组(unit doublet). 通过构造有限自动机(fini

lwip移植到stm32上-enc28j60,103mcu(2)

前面小玩了一下ucos和lwip,但是都还不是真正的网络多任务,真正的网络多任务应该是什么样子的呢?应该是有一个专门的任务负责网络的通讯,他负责将数据发送出去,将数据接收回来,而其他的需要用到网络的任务与这个任务通讯,这才是能够极大提升网络效率的办法,lwip支持这种办法,不过移植起来就比较磨人了 首先第一步是修改lwip的配置文件,如下 #define NO_SYS 0 //使用UCOS操作系统 改了这个之后会发现卧槽一下子缺了好多东西,实际上是这样,lwip使用一系列的宏和方法封装了一个操作

RT-Thread-v2.0.0移植到STM32及驱动LCD和测温

先简单记录下安装: 1. 安装 sourcery工具链或 keil都行 安装python 和 scons: 配置这些工具路径以及RT-Thread源码路径RTT_ROOT,添加到环境变量 2. 开始编译 bsp\stm32f10x工程,在编译以前需要稍微修改如下几个文件 rtconfig.py 此文件必须修改 rtconfig.h 此文件用于裁剪rt-thread(根据需要修改) 比如使用 keil工具来编译: 修改rtconfig.py, CROSS_TOOL='keil' EXEC_PATH

ucos实时操作系统学习笔记——操作系统在STM32的移植

使用ucos实时操作系统是在上学的时候,导师科研项目中.那时候就是网上找到操作系统移植教程以及应用教程依葫芦画瓢,功能实现也就罢了,没有很深入的去研究过这个东西.后来工作了,闲来无聊就研究了一下这个只有几千行代码的操作系统,也没所有的代码都看,只是看了其中部分内容.自己还自不量力的尝试着去写过简单的操作系统,最后写着写着就被带到了ucos的设计思路上了,后来干脆就“copy”代码了,虽说对操作系统内核的理解有很大的帮助,但是很是惭愧啊,智力不够,对操作系统内核的设计者更加仰慕,O(∩_∩)O哈哈

(笔记)CanOpen协议【CanFestival】移植方法 支持VC、QT、STM32

转自http://bbs.21ic.com/icview-878522-1-1.html 前段时间学习了CanOpen协议,到网上下载的CanFestival3-10源码,移植到VC.QT.STM32等平台,由于网上的资源较少,走了不少弯路,移植好使用过程中才逐渐暴露出各种问题,比如OD字符串传输.心跳时间不准确等等,现在已经解决了遇到的所有问题,移植出来的工程能够完好支持CanOpen协议,花了点时间,整理出一个简单易用的移植方法说明,也写了一些比较实用的调试工具,本来还想整理SDO.PDO.

解释器和编译器,编译型原理和解释型原理

解释器: 解释器(英语:Interpreter),又译为直译器,是一种电脑程序,能够把高级编程语言一行一行直接转译运行.解释器不会一次把整个程序转译出来,只像一位“中间人”,每次运行程序时都要先转成另一种语言再作运行,因此解释器的程序运行速度比较缓慢.它每转译一行程序叙述就立刻运行,然后再转译下一行,再运行,如此不停地进行下去. 解释器运行程序的方法有: 1.直接运行高级编程语言 (如 Shell 自带的解释器) 2.转换高级编程语言码到一些有效率的字节码 (Bytecode),并运行这些字节码