summaryrefslogtreecommitdiffstats
path: root/src/tools/pic30/pic30_generator.cpp
blob: 119c466058a6903e2c1875ff8a1d9195cf8e0fbe (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
/***************************************************************************
 *   Copyright (C) 2006-2007 Nicolas Hadacek <[email protected]>             *
 *                                                                         *
 *   This program is free software; you can redistribute it and/or modify  *
 *   it under the terms of the GNU General Public License as published by  *
 *   the Free Software Foundation; either version 2 of the License, or     *
 *   (at your option) any later version.                                   *
 ***************************************************************************/
#include "pic30_generator.h"

#include "devices/pic/pic/pic_memory.h"

SourceLine::List PIC30::SourceGenerator::configLines(PURL::ToolType type, const Device::Memory &memory, bool &ok) const
{
  const Pic::Memory &pmemory = static_cast<const Pic::Memory &>(memory);
  const Pic::Data &data = pmemory.device();
  const Pic::Config &config = data.config();
  SourceLine::List lines;
  for (uint i=0; i<data.nbWords(Pic::MemoryRangeType::Config); i++) {
    const Pic::Config::Word &cword = config._words[i];
    TQStringList cnames = SourceLine::configNames(Pic::ConfigNameType::Default, pmemory, i, ok);
    if ( cnames.isEmpty() ) continue;
    TQString code;
    if ( type==PURL::ToolType::Assembler ) code += "config __" + cword.name + ", ";
    else code += "_" + cword.name + "(";
    code += cnames.join(" & ");
    if ( type==PURL::ToolType::Compiler ) code += ");";
    lines.appendNotIndentedCode(code);
  }
  return lines;
}

SourceLine::List PIC30::SourceGenerator::includeLines(PURL::ToolType type, const Device::Data &data) const
{
  SourceLine::List lines;
  if ( type==PURL::ToolType::Assembler ) lines.appendIndentedCode(".include \"p" + data.name().lower() + ".inc\"");
  else lines.appendNotIndentedCode("#include <p" + data.name().lower() + ".h>");
  return lines;
}

SourceLine::List PIC30::SourceGenerator::sourceFileContent(PURL::ToolType type, const Device::Data &, bool &ok) const
{
  SourceLine::List lines;
  if ( type==PURL::ToolType::Assembler ) {
    lines.appendTitle(i18n("Global declarations"));
    lines.appendIndentedCode(".global _wreg_init");
    lines.appendIndentedCode(".global __reset", i18n("label for code start"));
    lines.appendIndentedCode(".global __T1Interrrupt", i18n("declare Timer1 ISR"));
    lines.appendEmpty();
    lines.appendSeparator();
    lines.appendTitle(i18n("Constants in program space"));
    lines.appendIndentedCode(".section .constbuffer, code");
    lines.appendIndentedCode(".palign 2");
    lines.appendNotIndentedCode("const1:");
    lines.appendIndentedCode(".hword 0x0001, 0x0002, 0X0003");
    lines.appendEmpty();
    lines.appendTitle(i18n("Unitialized variables in X-space"));
    lines.appendIndentedCode(".section .xbss, bss, xmemory");
    lines.appendNotIndentedCode("x_var1: .space 4", i18n("allocating 4 bytes"));
    lines.appendEmpty();
    lines.appendTitle(i18n("Unitialized variables in Y-space"));
    lines.appendIndentedCode(".section .ybss, bss, ymemory");
    lines.appendNotIndentedCode("y_var1: .space 2", i18n("allocating 2 bytes"));
    lines.appendEmpty();
    lines.appendTitle(i18n("Unitialized variables in near data memory"));
    lines.appendIndentedCode(".section .nbss, bss, near");
    lines.appendNotIndentedCode("var1: .space 4", i18n("allocating 4 bytes"));
    lines.appendEmpty();
    lines.appendSeparator();
    lines.appendTitle("Code section");
    lines.appendNotIndentedCode(".text");
    lines.appendNotIndentedCode("__reset:");
    lines.appendIndentedCode("MOV #__SP_init, W15", i18n("initialize stack pointer"));
    lines.appendIndentedCode("MOV #__SPLIM_init, W0", i18n("initialize stack pointer limit register"));
    lines.appendIndentedCode("MOV W0, SPLIM");
    lines.appendIndentedCode("NOP", i18n("nop after SPLIM initialization"));
    lines.appendIndentedCode("CALL _wreg_init", i18n("call _wreg_init subroutine"));
    lines.appendEmpty();
    lines.appendIndentedCode(TQString(), "<<" + i18n("insert code") + ">>");
    lines.appendEmpty();
    lines.appendNotIndentedCode("done:");
    lines.appendIndentedCode("BRA done", i18n("loop forever"));
    lines.appendEmpty();
    lines.appendSeparator();
    lines.appendTitle(i18n("Subroutine to initialize W registers to 0x0000"));
    lines.appendNotIndentedCode("_wreg_init:");
    lines.appendIndentedCode("CLR W0");
    lines.appendIndentedCode("MOV W0, W14");
    lines.appendIndentedCode("REPEAT #12");
    lines.appendIndentedCode("MOV W0, [++W14]");
    lines.appendIndentedCode("CLR W14");
    lines.appendIndentedCode("RETURN");
    lines.appendEmpty();
    lines.appendSeparator();
    lines.appendTitle(i18n("Timer1 interrupt service routine"));
    lines.appendNotIndentedCode("__T1Interrupt:");
    lines.appendIndentedCode("PUSH.D W4", i18n("example of context saving (push W4 and W5)"));
    lines.appendIndentedCode(TQString(), "<<" + i18n("insert interrupt code") + ">>");
    lines.appendIndentedCode("BCLR IFS0, #T1IF", i18n("clear Timer1 interrupt flag status bit"));
    lines.appendIndentedCode("POP.D W4", i18n("restore context from stack"));
    lines.appendIndentedCode("RETFIE");
    lines.appendEmpty();
    lines.appendSeparator();
    lines.appendTitle(i18n("End of all code sections"));
    lines.appendNotIndentedCode(".end");
  } else {
    // #### TODO
    ok = false;
  }
  return lines;
}