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
20 * Jürg Billeter <j@bitron.ch>
26 * Represents a writer to write C source files.
28 public class Vala
.CCodeWriter
{
30 * Specifies the file to be written.
32 public string filename
{ get; set; }
35 * Specifies the source file used to generate this one.
37 private string source_filename
;
40 * Specifies whether to emit line directives.
42 public bool line_directives
{ get; set; }
45 * Specifies whether the output stream is at the beginning of a line.
51 static GLib
.Regex fix_indent_regex
;
53 private string temp_filename
;
54 private bool file_exists
;
56 private FileStream? stream
;
59 private int current_line_number
= 1;
60 private bool using_line_directive
;
62 /* at begin of line */
63 private bool _bol
= true;
65 public CCodeWriter (string filename
, string? source_filename
= null) {
66 this
.filename
= filename
;
67 this
.source_filename
= source_filename
;
73 * @return true if the file has been opened successfully,
76 public bool open (bool write_version
) {
77 file_exists
= FileUtils
.test (filename
, FileTest
.EXISTS
);
79 temp_filename
= "%s.valatmp".printf (filename
);
80 stream
= FileStream
.open (temp_filename
, "w");
83 * File doesn't exist. In case of a particular destination (-d flag),
84 * check and create the directory structure.
86 var dirname
= Path
.get_dirname (filename
);
87 DirUtils
.create_with_parents (dirname
, 0755);
88 stream
= FileStream
.open (filename
, "w");
95 var opening
= write_version ?
96 "/* %s generated by valac %s, the Vala compiler".printf (Path
.get_basename (filename
), Config
.BUILD_VERSION
) :
97 "/* %s generated by valac, the Vala compiler".printf (Path
.get_basename (filename
));
98 write_string (opening
);
100 // Write the file name if known
101 if (source_filename
!= null) {
103 write_string (" * generated from %s".printf (Path
.get_basename (source_filename
)));
106 write_string (", do not modify */");
116 public void close () {
123 var old_file
= new
MappedFile (filename
, false);
124 var new_file
= new
MappedFile (temp_filename
, false);
125 var len
= old_file
.get_length ();
126 if (len
== new_file
.get_length ()) {
127 if (Memory
.cmp (old_file
.get_contents (), new_file
.get_contents (), len
) == 0) {
133 } catch (FileError e
) {
134 // assume changed if mmap comparison doesn't work
138 FileUtils
.rename (temp_filename
, filename
);
140 FileUtils
.unlink (temp_filename
);
141 if (source_filename
!= null) {
142 var stats
= Stat (source_filename
);
143 var target_stats
= Stat (filename
);
144 if (stats
.st_mtime
>= target_stats
.st_mtime
) {
145 UTimBuf timebuf
= { stats
.st_atime
+ 1, stats
.st_mtime
+ 1 };
146 FileUtils
.utime (filename
, timebuf
);
154 * Writes tabs according to the current indent level.
156 public void write_indent (CCodeLineDirective? line
= null) {
157 if (line_directives
) {
160 using_line_directive
= true;
161 } else if (using_line_directive
) {
162 // no corresponding Vala line, emit line directive for C line
163 write_string ("#line %d \"%s\"".printf (current_line_number
+ 1, Path
.get_basename (filename
)));
165 using_line_directive
= false;
173 stream
.puts (string.nfill (indent
, '\t'));
180 public void write_nspaces (uint n
) {
181 stream
.puts (string.nfill (n
, ' '));
185 * Writes the specified string.
189 public void write_string (string s
) {
197 public void write_newline () {
199 current_line_number
++;
204 * Opens a new block, increasing the indent level.
206 public void write_begin_block () {
218 * Closes the current block, decreasing the indent level.
220 public void write_end_block () {
229 * Writes the specified text as comment.
231 * @param text the comment text
233 public void write_comment (string text
) {
239 // discard tabs at beginning of line
240 if (fix_indent_regex
== null)
241 fix_indent_regex
= new GLib
.Regex ("^\t+");;
243 foreach (unowned
string line
in text
.split ("\n")) {
250 var lineparts
= fix_indent_regex
.replace_literal (line
, -1, 0, "").split ("*/");
252 for (int i
= 0; lineparts
[i
] != null; i
++) {
253 stream
.puts (lineparts
[i
]);
254 if (lineparts
[i
+1] != null) {
261 } catch (RegexError e
) {