Release 0.7.8
[vala-lang.git] / ccode / valaccodewriter.vala
blobff0677bceba4239b83d63c79ccfca05a32aaba78
1 /* valaccodewriter.vala
3 * Copyright (C) 2006-2009 Jürg Billeter
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19 * Author:
20 * Jürg Billeter <j@bitron.ch>
23 using GLib;
25 /**
26 * Represents a writer to write C source files.
28 public class Vala.CCodeWriter {
29 /**
30 * Specifies the file to be written.
32 public string filename { get; set; }
34 /**
35 * Specifies the source file used to generate this one.
37 private string source_filename;
39 /**
40 * Specifies whether to emit line directives.
42 public bool line_directives { get; set; }
44 /**
45 * Specifies whether the output stream is at the beginning of a line.
47 public bool bol {
48 get { return _bol; }
51 private string temp_filename;
52 private bool file_exists;
54 private FileStream stream;
56 private int indent;
57 private int current_line_number = 1;
58 private bool using_line_directive;
60 /* at begin of line */
61 private bool _bol = true;
63 public CCodeWriter (string filename, string? source_filename = null) {
64 this.filename = filename;
65 this.source_filename = source_filename;
68 /**
69 * Opens the file.
71 * @return true if the file has been opened successfully,
72 * false otherwise
74 public bool open () {
75 file_exists = FileUtils.test (filename, FileTest.EXISTS);
76 if (file_exists) {
77 temp_filename = "%s.valatmp".printf (filename);
78 stream = FileStream.open (temp_filename, "w");
79 } else {
80 stream = FileStream.open (filename, "w");
83 write_string ("/* %s generated by valac, the Vala compiler".printf (Path.get_basename (filename)));
85 // Write the file name if known
86 if (source_filename != null) {
87 write_newline ();
88 write_string (" * generated from %s".printf (Path.get_basename (source_filename)));
91 write_string (", do not modify */");
92 write_newline ();
93 write_newline ();
95 return (stream != null);
98 /**
99 * Closes the file.
101 public void close () {
102 stream = null;
104 if (file_exists) {
105 var changed = true;
107 try {
108 var old_file = new MappedFile (filename, false);
109 var new_file = new MappedFile (temp_filename, false);
110 var len = old_file.get_length ();
111 if (len == new_file.get_length ()) {
112 if (Memory.cmp (old_file.get_contents (), new_file.get_contents (), len) == 0) {
113 changed = false;
116 old_file = null;
117 new_file = null;
118 } catch (FileError e) {
119 // assume changed if mmap comparison doesn't work
122 if (changed) {
123 FileUtils.rename (temp_filename, filename);
124 } else {
125 FileUtils.unlink (temp_filename);
131 * Writes tabs according to the current indent level.
133 public void write_indent (CCodeLineDirective? line = null) {
134 if (line_directives) {
135 if (line != null) {
136 line.write (this);
137 using_line_directive = true;
138 } else if (using_line_directive) {
139 // no corresponding Vala line, emit line directive for C line
140 write_string ("#line %d \"%s\"".printf (current_line_number + 1, Path.get_basename (filename)));
141 write_newline ();
142 using_line_directive = false;
146 if (!bol) {
147 write_newline ();
150 for (int i = 0; i < indent; i++) {
151 stream.putc ('\t');
154 _bol = false;
158 * Writes the specified string.
160 * @param s a string
162 public void write_string (string s) {
163 stream.puts (s);
164 _bol = false;
168 * Writes a newline.
170 public void write_newline () {
171 stream.putc ('\n');
172 current_line_number++;
173 _bol = true;
177 * Opens a new block, increasing the indent level.
179 public void write_begin_block () {
180 if (!bol) {
181 stream.putc (' ');
182 } else {
183 write_indent ();
185 stream.putc ('{');
186 write_newline ();
187 indent++;
191 * Closes the current block, decreasing the indent level.
193 public void write_end_block () {
194 assert (indent > 0);
196 indent--;
197 write_indent ();
198 stream.putc ('}');
202 * Writes the specified text as comment.
204 * @param text the comment text
206 public void write_comment (string text) {
207 write_indent ();
208 stream.puts ("/*");
209 bool first = true;
211 /* separate declaration due to missing memory management in foreach statements */
212 var lines = text.split ("\n");
214 foreach (string line in lines) {
215 if (!first) {
216 write_indent ();
217 } else {
218 first = false;
221 var lineparts = line.split ("*/");
223 for (int i = 0; lineparts[i] != null; i++) {
224 stream.puts (lineparts[i]);
225 if (lineparts[i+1] != null) {
226 stream.puts ("* /");
230 stream.puts ("*/");
231 write_newline ();