OpenCores
URL https://opencores.org/ocsvn/neo430/neo430/trunk

Subversion Repositories neo430

[/] [neo430/] [trunk/] [neo430/] [sw/] [lib/] [neo430/] [source/] [neo430_cpu.c] - Rev 198

Compare with Previous | Blame | View Log

// #################################################################################################
// #  < neo430_cpu.c - CPU helper functions >                                                      #
// # ********************************************************************************************* #
// # BSD 3-Clause License                                                                          #
// #                                                                                               #
// # Copyright (c) 2020, Stephan Nolting. All rights reserved.                                     #
// #                                                                                               #
// # Redistribution and use in source and binary forms, with or without modification, are          #
// # permitted provided that the following conditions are met:                                     #
// #                                                                                               #
// # 1. Redistributions of source code must retain the above copyright notice, this list of        #
// #    conditions and the following disclaimer.                                                   #
// #                                                                                               #
// # 2. 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.                                                            #
// #                                                                                               #
// # 3. Neither the name of the copyright holder nor the names of its contributors may be used to  #
// #    endorse or promote products derived from this software without specific prior written      #
// #    permission.                                                                                #
// #                                                                                               #
// # 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 HOLDER 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.                                                            #
// # ********************************************************************************************* #
// # The NEO430 Processor - https://github.com/stnolting/neo430                                    #
// #################################################################################################
 
#include "neo430.h"
#include "neo430_cpu.h"
 
// Private variables
static uint16_t __neo430_sreg __attribute__((unused)); // do not ouput a warning when this variable is unused
 
 
/* ------------------------------------------------------------
 * INFO Beginning of critical section (store SREG and disable interrupts)
 * ------------------------------------------------------------ */
void neo430_critical_start(void) {
 
  register uint16_t d;
  asm volatile ("mov r2, %0" : "=r" (d));
  __neo430_sreg = d; // store current SREG
 
  asm volatile ("dint");
  asm volatile ("nop");
}
 
 
/* ------------------------------------------------------------
 * INFO End of critical section (restore original SREG)
 * ------------------------------------------------------------ */
void neo430_critical_end(void) {
 
  register uint16_t r = __neo430_sreg;
  asm volatile ("mov %0, r2" : : "r" (r));
}
 
 
/* ------------------------------------------------------------
 * INFO Enable global interrupt flag
 * ------------------------------------------------------------ */
void neo430_eint(void){
 
  asm volatile ("eint");
  asm volatile ("nop");
}
 
 
/* ------------------------------------------------------------
 * INFO Disable global interrupt flag
 * ------------------------------------------------------------ */
void neo430_dint(void){
 
  asm volatile ("dint");
  asm volatile ("nop");
}
 
 
/* ------------------------------------------------------------
 * INFO Read stack pointer (for debugging only)
 * RETURN current stack pointer
 * ------------------------------------------------------------ */
uint16_t neo430_get_sp(void){
 
  register uint16_t d;
  asm volatile ("mov r1, %0" : "=r" (d));
  uint16_t r = d;
  return r;
}
 
 
/* ------------------------------------------------------------
 * INFO Read status register
 * RETURN current status register
 * ------------------------------------------------------------ */
uint16_t neo430_get_sreg(void){
 
  register uint16_t d;
  asm volatile ("mov r2, %0" : "=r" (d));
  uint16_t r = d;
  return r;
}
 
 
/* ------------------------------------------------------------
 * INFO Set status register
 * PARAM d new value for status register
 * ------------------------------------------------------------ */
void neo430_set_sreg(uint16_t d){
 
  register uint16_t r = d;
  asm volatile ("mov %0, r2" : : "r" (r));
}
 
 
/* ------------------------------------------------------------
 * INFO Get parity of value
 * WARNING MAKE SURE THIS OPTION IS SYNTHESIZED (package switch "use_ext_alu_c")!!!
 * PARAM d input value
 * RETURN Resulting parity (1=even number of 1s, 0=odd number of 1s)
 * ------------------------------------------------------------ */
uint16_t neo430_get_parity(uint16_t d){
 
  register uint16_t r = d;
  asm volatile ("mov %0, %0" : "=r" (r) : "r" (r)); // just get value through alu
 
  if (neo430_get_sreg() & (1<<P_FLAG)) // get parity flag from SR
    return 1;
  else
    return 0;
}
 
 
/* ------------------------------------------------------------
 * INFO Set CPU to sleep mode
 * ------------------------------------------------------------ */
void neo430_sleep(void){
 
  asm volatile ("bis %0, r2" : : "i" (1<<S_FLAG));
}
 
 
/* ------------------------------------------------------------
 * INFO Clear CPU pending IRQ buffer
 * ------------------------------------------------------------ */
void neo430_clear_irq_buffer(void){
 
  asm volatile ("bis %0, r2" : : "i" (1<<Q_FLAG));
  // no need to reset the flag as it automatically clears again
}
 
 
/* ------------------------------------------------------------
 * INFO Simple wait function
 * PARAM Amount of ~2^16 machine cycles to wait
 * ------------------------------------------------------------ */
void neo430_cpu_delay(uint16_t t) {
 
  register uint16_t i = 0;
  while (t--) {
    for (i=0; i<0xFFFF; i++)
      asm volatile ("nop");
  }
}
 
 
/* ------------------------------------------------------------
 * INFO Waits <ms> microseconds (not very precise!)
 * PARAM ms time in microseconds to wait
 * ------------------------------------------------------------ */
void neo430_cpu_delay_ms(uint16_t ms) {
 
  // empirical ;)
  uint32_t a = ((uint32_t)CLOCKSPEED_HI) << 1;
  register uint32_t cnt = a * (uint32_t)ms;
 
  while(cnt--) {
    asm volatile ("nop");
  }
}
 
 
/* ------------------------------------------------------------
 * INFO Perform a soft reset by jumping to beginning of IMEM
 * ------------------------------------------------------------ */
void neo430_soft_reset(void) {
 
  asm volatile ("mov #0x0000, r0");
}
 
 
/* ------------------------------------------------------------
 * INFO Jump to address
 * PARAM Destination address
 * ------------------------------------------------------------ */
void neo430_jump_address(uint16_t addr) {
 
  register uint16_t r = addr;
  asm volatile ("mov %0, r0" : : "r" (r));
}
 
 
/* ------------------------------------------------------------
 * INFO Call address and save return address to stack
 * PARAM Destination address
 * ------------------------------------------------------------ */
void neo430_call_address(uint16_t addr) {
 
  register uint16_t r = addr;
  asm volatile ("call %0" : : "r" (r));
}
 
 
/* ------------------------------------------------------------
 * INFO Perform byte swap of 16-bit word (e.g., for endianness conversion)
 * PARAM 16-bit input word
 * RETURN 16-bit word with swapped bytes
 * ------------------------------------------------------------ */
uint16_t neo430_bswap(uint16_t a) {
 
  register uint16_t r = a;
  asm volatile ("swpb %0, %1" : "=r" (r) : "r" (r));
  return r;
}
 
 
/* ------------------------------------------------------------
 * INFO Combine two bytes into one word
 * PARAM hi will be put in result's high byte
 * PARAM lo will be put in result's low byte
 * RETURN 16-bit combined word
 * ------------------------------------------------------------ */
uint16_t neo430_combine_bytes(uint8_t hi, uint8_t lo) {
 
  register uint16_t r = neo430_bswap((uint16_t)hi);
  return r | (uint16_t)lo;
}
 
 
/* ------------------------------------------------------------
 * INFO Memory initialization (byte-wise)
 * PARAM dst: Byte-pointer to beginning of target memory space
 * PARAM data: Init data
 * PARAM num: Number of bytes to initialize
 * ------------------------------------------------------------ */
void neo430_memset(uint8_t *dst, uint8_t data, uint16_t num) {
 
  while (num--)
    *dst++ = data;
}
 
 
/* ------------------------------------------------------------
 * INFO Compare memory to memory
 * PARAM dst: Pointer to beginning of first memory space
 * PARAM src: Pointer to beginning of second memory space
 * PARAM num: Number of bytes to compare
 * RETURN 0 if src == dst
 * ------------------------------------------------------------ */
uint8_t neo430_memcmp(uint8_t *dst, uint8_t *src, uint16_t num) {
 
  while (num--) {
    if (*dst++ != *src++)
      return 1;
  }
  return 0;
}
 
 
/* ------------------------------------------------------------
 * INFO Copy memory space SRC to DST (byte by byte)
 * PARAM dst: Pointer to beginning destination memory space
 * PARAM src: Pointer to beginning source memory space
 * PARAM num: Number of bytes to copy
 * ------------------------------------------------------------ */
void neo430_memcpy(uint8_t *dst, uint8_t *src, uint16_t num) {
 
  while (num--)
    *dst++ = *src++;
}
 
 
/* ------------------------------------------------------------
 * INFO 16-bit bit reversal
 * PARAM input operand to be reversed
 * RETURN reversed bit pattern
 * ------------------------------------------------------------ */
uint16_t neo430_bit_rev16(uint16_t x) {
 
  register uint16_t z = x;
  register uint16_t y = 0;
  uint8_t i = 0;
 
  for (i=0; i<8; i++) { // two-times unrolled
    asm volatile ("rrc %[a], %[b]" : [b] "=r" (z) : "[b]" (z), [a] "r" (z));
    asm volatile ("rlc %[c], %[d]" : [d] "=r" (y) : "[d]" (y), [c] "r" (y));
 
    asm volatile ("rrc %[a], %[b]" : [b] "=r" (z) : "[b]" (z), [a] "r" (z));
    asm volatile ("rlc %[c], %[d]" : [d] "=r" (y) : "[d]" (y), [c] "r" (y));
  }
  return y;
}
 
 
/* ------------------------------------------------------------
 * INFO rotate word right by one position
 * PARAM input operand to be rotated
 * RETURN rotated result
 * ------------------------------------------------------------ */
uint16_t neo430_rotate_right_w(uint16_t x) {
 
  uint16_t tmp = x;
  asm volatile ("rrc.w %[b]" : [b] "=r" (tmp) : "[b]" (tmp)); // get carry flag
  asm volatile ("rrc.w %[b]" : [b] "=r" (x) : "[b]" (x)); // rotate input with according carry input
  return x;
}
 
 
/* ------------------------------------------------------------
 * INFO rotate word left by one position
 * PARAM input operand to be rotated
 * RETURN rotated result
 * ------------------------------------------------------------ */
uint16_t neo430_rotate_left_w(uint16_t x) {
 
  uint16_t tmp = x;
  asm volatile ("rlc.w %[b]" : [b] "=r" (tmp) : "[b]" (tmp)); // get carry flag
  asm volatile ("rlc.w %[b]" : [b] "=r" (x) : "[b]" (x)); // rotate input with according carry input
  return x;
}
 
 
/* ------------------------------------------------------------
 * INFO rotate byte right by one position
 * PARAM input operand to be rotated
 * RETURN rotated result
 * ------------------------------------------------------------ */
uint8_t neo430_rotate_right_b(uint8_t x) {
 
  uint8_t tmp = x;
  asm volatile ("rrc.b %[b]" : [b] "=r" (tmp) : "[b]" (tmp)); // get carry flag
  asm volatile ("rrc.b %[b]" : [b] "=r" (x) : "[b]" (x)); // rotate input with according carry input
  return x;
}
 
 
/* ------------------------------------------------------------
 * INFO rotate byte left by one position
 * PARAM input operand to be rotated
 * RETURN rotated result
 * ------------------------------------------------------------ */
uint8_t neo430_rotate_left_b(uint8_t x) {
 
  uint8_t tmp = x;
  asm volatile ("rlc.b %[b]" : [b] "=r" (tmp) : "[b]" (tmp)); // get carry flag
  asm volatile ("rlc.b %[b]" : [b] "=r" (x) : "[b]" (x)); // rotate input with according carry input
  return x;
}
 
 
/* ------------------------------------------------------------
 * INFO Pseudo-random number generator
 * RETURN 32-bit random data
 * ------------------------------------------------------------ */
uint32_t neo430_xorshift32(void) {
 
  static uint32_t x32 = 314159265;
 
  x32 ^= x32 << 13;
  x32 ^= x32 >> 17;
  x32 ^= x32 << 5;
 
  return x32;
}
 

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.