89763f52ec9185327ba7f17d63a6ffc14e349b6e
[mono.git] / mcs / class / Mono.CompilerServices.SymbolWriter / SourceMethodBuilder.cs
1 //
2 // SourceMethodBuilder.cs
3 //
4 // Authors:
5 //   Martin Baulig (martin@ximian.com)
6 //   Marek Safar (marek.safar@gmail.com)
7 //
8 // (C) 2002 Ximian, Inc.  http://www.ximian.com
9 // Copyright (C) 2012 Xamarin Inc (http://www.xamarin.com)
10 //
11 // Permission is hereby granted, free of charge, to any person obtaining
12 // a copy of this software and associated documentation files (the
13 // "Software"), to deal in the Software without restriction, including
14 // without limitation the rights to use, copy, modify, merge, publish,
15 // distribute, sublicense, and/or sell copies of the Software, and to
16 // permit persons to whom the Software is furnished to do so, subject to
17 // the following conditions:
18 // 
19 // The above copyright notice and this permission notice shall be
20 // included in all copies or substantial portions of the Software.
21 // 
22 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
23 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
24 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
25 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
26 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
27 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
28 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 //
30
31 using System;
32 using System.Collections.Generic;
33
34 namespace Mono.CompilerServices.SymbolWriter
35 {
36         public class SourceMethodBuilder
37         {
38                 List<LocalVariableEntry> _locals;
39                 List<CodeBlockEntry> _blocks;
40                 List<ScopeVariable> _scope_vars;
41                 Stack<CodeBlockEntry> _block_stack;
42                 readonly List<LineNumberEntry> method_lines;
43
44                 readonly ICompileUnit _comp_unit;
45                 readonly int ns_id;
46                 readonly IMethodDef method;
47
48                 public SourceMethodBuilder (ICompileUnit comp_unit)
49                 {
50                         this._comp_unit = comp_unit;
51                         method_lines = new List<LineNumberEntry> ();
52                 }
53
54                 public SourceMethodBuilder (ICompileUnit comp_unit, int ns_id, IMethodDef method)
55                         : this (comp_unit)
56                 {
57                         this.ns_id = ns_id;
58                         this.method = method;
59                 }
60
61                 public void MarkSequencePoint (int offset, SourceFileEntry file, int line, int column, bool is_hidden)
62                 {
63                         MarkSequencePoint (offset, file, line, column, -1, -1, is_hidden);
64                 }
65
66                 public void MarkSequencePoint (int offset, SourceFileEntry file, int line, int column, int end_line, int end_column, bool is_hidden)
67                 {
68                         int file_idx = file != null ? file.Index : 0;
69                         var lne = new LineNumberEntry (file_idx, line, column, end_line, end_column, offset, is_hidden);
70
71                         if (method_lines.Count > 0) {
72                                 var prev = method_lines[method_lines.Count - 1];
73
74                                 //
75                                 // Same offset cannot be used for multiple lines
76                                 // 
77                                 if (prev.Offset == offset) {
78                                         //
79                                         // Use the new location because debugger will adjust
80                                         // the breakpoint to next line with sequence point
81                                         //
82                                         if (LineNumberEntry.LocationComparer.Default.Compare (lne, prev) > 0)
83                                                 method_lines[method_lines.Count - 1] = lne;
84
85                                         return;
86                                 }
87                         }
88
89                         method_lines.Add (lne);
90                 }
91
92                 public void StartBlock (CodeBlockEntry.Type type, int start_offset)
93                 {
94                         StartBlock (type, start_offset, _blocks == null ? 1 : _blocks.Count + 1);
95                 }
96
97                 public void StartBlock (CodeBlockEntry.Type type, int start_offset, int scopeIndex)
98                 {
99                         if (_block_stack == null) {
100                                 _block_stack = new Stack<CodeBlockEntry> ();
101                         }
102                         
103                         if (_blocks == null)
104                                 _blocks = new List<CodeBlockEntry> ();
105
106                         int parent = CurrentBlock != null ? CurrentBlock.Index : -1;
107
108                         CodeBlockEntry block = new CodeBlockEntry (
109                                 scopeIndex, parent, type, start_offset);
110
111                         _block_stack.Push (block);
112                         _blocks.Add (block);
113                 }
114
115                 public void EndBlock (int end_offset)
116                 {
117                         CodeBlockEntry block = (CodeBlockEntry) _block_stack.Pop ();
118                         block.Close (end_offset);
119                 }
120
121                 public CodeBlockEntry[] Blocks {
122                         get {
123                                 if (_blocks == null)
124                                         return new CodeBlockEntry [0];
125
126                                 CodeBlockEntry[] retval = new CodeBlockEntry [_blocks.Count];
127                                 _blocks.CopyTo (retval, 0);
128                                 return retval;
129                         }
130                 }
131
132                 public CodeBlockEntry CurrentBlock {
133                         get {
134                                 if ((_block_stack != null) && (_block_stack.Count > 0))
135                                         return (CodeBlockEntry) _block_stack.Peek ();
136                                 else
137                                         return null;
138                         }
139                 }
140
141                 public LocalVariableEntry[] Locals {
142                         get {
143                                 if (_locals == null)
144                                         return new LocalVariableEntry [0];
145                                 else {
146                                         return _locals.ToArray ();
147                                 }
148                         }
149                 }
150
151                 public ICompileUnit SourceFile {
152                         get {
153                                 return _comp_unit;
154                         }
155                 }
156
157                 public void AddLocal (int index, string name)
158                 {
159                         if (_locals == null)
160                                 _locals = new List<LocalVariableEntry> ();
161                         int block_idx = CurrentBlock != null ? CurrentBlock.Index : 0;
162                         _locals.Add (new LocalVariableEntry (index, name, block_idx));
163                 }
164
165                 public ScopeVariable[] ScopeVariables {
166                         get {
167                                 if (_scope_vars == null)
168                                         return new ScopeVariable [0];
169
170                                 return _scope_vars.ToArray ();
171                         }
172                 }
173
174                 public void AddScopeVariable (int scope, int index)
175                 {
176                         if (_scope_vars == null)
177                                 _scope_vars = new List<ScopeVariable> ();
178                         _scope_vars.Add (
179                                 new ScopeVariable (scope, index));
180                 }
181
182                 public void DefineMethod (MonoSymbolFile file)
183                 {
184                         DefineMethod (file, method.Token);
185                 }
186
187                 public void DefineMethod (MonoSymbolFile file, int token)
188                 {
189                         var blocks = Blocks;
190
191                         //
192                         // When index is provided by user it can be inserted in
193                         // any order but mdb format does not store its value. It
194                         // uses store order instead as the index.
195                         //
196                         Array.Sort (blocks, (x, y) => x.Index.CompareTo (y.Index));
197
198                         var entry = new MethodEntry (
199                                 file, _comp_unit.Entry, token, ScopeVariables,
200                                 Locals, method_lines.ToArray (), blocks, null, MethodEntry.Flags.ColumnsInfoIncluded, ns_id);
201
202                         file.AddMethod (entry);
203                 }
204         }
205 }