src/dmx_frame.c
author Tero Marttila <terom@paivola.fi>
Sun, 06 Apr 2014 18:25:55 +0300
changeset 62 2d68a76322cb
permissions -rw-r--r--
hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
62
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     1
/*
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     2
 * Per-frame timing-critical bit-banging.
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     3
 *
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     4
 * Shamelessly stolen from
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     5
 *  http://code.google.com/p/tinkerit/source/browse/trunk/DmxSimple/DmxSimple.cpp
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     6
 *
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     7
 *  Copyright (c) 2008-2009 Peter Knight, Tinker.it!
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     8
 *
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
     9
 * Consumes approx. 11 * 64 ~= 705 clock cycles
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    10
 * Leaves the line idle (high) on return.
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    11
 */
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    12
void dmx_frame (volatile byte value)
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    13
{
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    14
  uint8_t bitCount, delCount;
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    15
  __asm__ volatile (
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    16
    "cli\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    17
    "ld __tmp_reg__,%a[dmxPort]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    18
    "and __tmp_reg__,%[outMask]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    19
    "st %a[dmxPort],__tmp_reg__\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    20
    "ldi %[bitCount],11\n" // 11 bit intervals per transmitted byte
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    21
    "rjmp bitLoop%=\n"     // Delay 2 clock cycles.
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    22
  "bitLoop%=:\n"\
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    23
    "ldi %[delCount],%[delCountVal]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    24
  "delLoop%=:\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    25
    "nop\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    26
    "dec %[delCount]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    27
    "brne delLoop%=\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    28
    "ld __tmp_reg__,%a[dmxPort]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    29
    "and __tmp_reg__,%[outMask]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    30
    "sec\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    31
    "ror %[value]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    32
    "brcc sendzero%=\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    33
    "or __tmp_reg__,%[outBit]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    34
  "sendzero%=:\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    35
    "st %a[dmxPort],__tmp_reg__\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    36
    "dec %[bitCount]\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    37
    "brne bitLoop%=\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    38
    "sei\n"
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    39
    :
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    40
      [bitCount] "=&d" (bitCount),
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    41
      [delCount] "=&d" (delCount)
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    42
    :
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    43
      [dmxPort] "e" (&DMX_PORT),
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    44
      [outMask] "r" (~(1 << DMX_DATA)),
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    45
      [outBit] "r" ((1 << DMX_DATA)),
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    46
      [delCountVal] "M" (F_CPU/1000000-3),
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    47
      [value] "r" (value)
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    48
  );
2d68a76322cb hello-dmx: working basic dmx output using DmxSimple's frame-timing inline assembler code
Tero Marttila <terom@paivola.fi>
parents:
diff changeset
    49
}