Coverage Report

Created: 2026-06-02 06:36

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/Zend/zend_highlight.c
Line
Count
Source
1
/*
2
   +----------------------------------------------------------------------+
3
   | Zend Engine                                                          |
4
   +----------------------------------------------------------------------+
5
   | Copyright © Zend Technologies Ltd., a subsidiary company of          |
6
   |     Perforce Software, Inc., and Contributors.                       |
7
   +----------------------------------------------------------------------+
8
   | This source file is subject to the Modified BSD License that is      |
9
   | bundled with this package in the file LICENSE, and is available      |
10
   | through the World Wide Web at <https://www.php.net/license/>.        |
11
   |                                                                      |
12
   | SPDX-License-Identifier: BSD-3-Clause                                |
13
   +----------------------------------------------------------------------+
14
   | Authors: Andi Gutmans <andi@php.net>                                 |
15
   |          Zeev Suraski <zeev@php.net>                                 |
16
   +----------------------------------------------------------------------+
17
*/
18
19
#include "zend.h"
20
#include <zend_language_parser.h>
21
#include "zend_compile.h"
22
#include "zend_highlight.h"
23
#include "zend_ptr_stack.h"
24
#include "zend_globals.h"
25
#include "zend_exceptions.h"
26
27
ZEND_API void zend_html_putc(char c)
28
1.25M
{
29
1.25M
  switch (c) {
30
81.6k
    case '<':
31
81.6k
      ZEND_PUTS("&lt;");
32
81.6k
      break;
33
1.08k
    case '>':
34
1.08k
      ZEND_PUTS("&gt;");
35
1.08k
      break;
36
563
    case '&':
37
563
      ZEND_PUTS("&amp;");
38
563
      break;
39
2.80k
    case '\t':
40
2.80k
      ZEND_PUTS("    ");
41
2.80k
      break;
42
1.16M
    default:
43
1.16M
      ZEND_PUTC(c);
44
1.16M
      break;
45
1.25M
  }
46
1.25M
}
47
48
49
ZEND_API void zend_html_puts(const char *s, size_t len)
50
431k
{
51
431k
  const unsigned char *ptr = (const unsigned char*)s, *end = ptr + len;
52
431k
  unsigned char *filtered = NULL;
53
431k
  size_t filtered_len;
54
55
431k
  if (LANG_SCNG(output_filter)) {
56
0
    LANG_SCNG(output_filter)(&filtered, &filtered_len, ptr, len);
57
0
    ptr = filtered;
58
0
    end = filtered + filtered_len;
59
0
  }
60
61
1.65M
  while (ptr<end) {
62
1.22M
    if (*ptr==' ') {
63
96.1k
      do {
64
96.1k
        zend_html_putc(*ptr);
65
96.1k
      } while ((++ptr < end) && (*ptr==' '));
66
1.15M
    } else {
67
1.15M
      zend_html_putc(*ptr++);
68
1.15M
    }
69
1.22M
  }
70
71
431k
  if (LANG_SCNG(output_filter)) {
72
0
    efree(filtered);
73
0
  }
74
431k
}
75
76
77
ZEND_API void zend_highlight(zend_syntax_highlighter_ini *syntax_highlighter_ini)
78
3.40k
{
79
3.40k
  zval token;
80
3.40k
  int token_type;
81
3.40k
  const char *last_color = syntax_highlighter_ini->highlight_html;
82
3.40k
  const char *next_color;
83
84
3.40k
  zend_printf("<pre><code style=\"color: %s\">", last_color);
85
  /* highlight stuff coming back from zendlex() */
86
434k
  while ((token_type=lex_scan(&token, NULL))) {
87
431k
    switch (token_type) {
88
3.45k
      case T_INLINE_HTML:
89
3.45k
        next_color = syntax_highlighter_ini->highlight_html;
90
3.45k
        break;
91
1.54k
      case T_COMMENT:
92
1.54k
      case T_DOC_COMMENT:
93
1.54k
        next_color = syntax_highlighter_ini->highlight_comment;
94
1.54k
        break;
95
3.10k
      case T_OPEN_TAG:
96
3.26k
      case T_OPEN_TAG_WITH_ECHO:
97
3.40k
      case T_CLOSE_TAG:
98
3.40k
      case T_LINE:
99
3.40k
      case T_FILE:
100
3.40k
      case T_DIR:
101
3.40k
      case T_TRAIT_C:
102
3.40k
      case T_METHOD_C:
103
3.40k
      case T_FUNC_C:
104
3.40k
      case T_PROPERTY_C:
105
3.40k
      case T_NS_C:
106
3.40k
      case T_CLASS_C:
107
3.40k
        next_color = syntax_highlighter_ini->highlight_default;
108
3.40k
        break;
109
232
      case '"':
110
2.59k
      case T_ENCAPSED_AND_WHITESPACE:
111
7.92k
      case T_CONSTANT_ENCAPSED_STRING:
112
7.92k
        next_color = syntax_highlighter_ini->highlight_string;
113
7.92k
        break;
114
61.4k
      case T_WHITESPACE:
115
61.4k
        zend_html_puts((char*)LANG_SCNG(yy_text), LANG_SCNG(yy_leng));  /* no color needed */
116
61.4k
        ZVAL_UNDEF(&token);
117
61.4k
        continue;
118
0
        break;
119
353k
      default:
120
353k
        if (Z_TYPE(token) == IS_UNDEF) {
121
255k
          next_color = syntax_highlighter_ini->highlight_keyword;
122
255k
        } else {
123
97.8k
          next_color = syntax_highlighter_ini->highlight_default;
124
97.8k
        }
125
353k
        break;
126
431k
    }
127
128
369k
    if (last_color != next_color) {
129
171k
      if (last_color != syntax_highlighter_ini->highlight_html) {
130
167k
        zend_printf("</span>");
131
167k
      }
132
171k
      last_color = next_color;
133
171k
      if (last_color != syntax_highlighter_ini->highlight_html) {
134
170k
        zend_printf("<span style=\"color: %s\">", last_color);
135
170k
      }
136
171k
    }
137
138
369k
    zend_html_puts((char*)LANG_SCNG(yy_text), LANG_SCNG(yy_leng));
139
140
369k
    if (Z_TYPE(token) == IS_STRING) {
141
82.2k
      switch (token_type) {
142
0
        case T_OPEN_TAG:
143
0
        case T_OPEN_TAG_WITH_ECHO:
144
0
        case T_CLOSE_TAG:
145
0
        case T_WHITESPACE:
146
0
        case T_COMMENT:
147
0
        case T_DOC_COMMENT:
148
0
          break;
149
82.2k
        default:
150
82.2k
          zval_ptr_dtor_str(&token);
151
82.2k
          break;
152
82.2k
      }
153
82.2k
    }
154
369k
    ZVAL_UNDEF(&token);
155
369k
  }
156
157
3.40k
  if (last_color != syntax_highlighter_ini->highlight_html) {
158
3.11k
    zend_printf("</span>");
159
3.11k
  }
160
3.40k
  zend_printf("</code></pre>");
161
162
  /* Discard parse errors thrown during tokenization */
163
3.40k
  zend_clear_exception();
164
3.40k
}
165
166
ZEND_API void zend_strip(void)
167
0
{
168
0
  zval token;
169
0
  int token_type;
170
0
  int prev_space = 0;
171
172
0
  while ((token_type=lex_scan(&token, NULL))) {
173
0
    switch (token_type) {
174
0
      case T_WHITESPACE:
175
0
        if (!prev_space) {
176
0
          zend_write(" ", sizeof(" ") - 1);
177
0
          prev_space = 1;
178
0
        }
179
0
        ZEND_FALLTHROUGH;
180
0
      case T_COMMENT:
181
0
      case T_DOC_COMMENT:
182
0
        ZVAL_UNDEF(&token);
183
0
        continue;
184
185
0
      case T_END_HEREDOC:
186
0
        zend_write((char*)LANG_SCNG(yy_text), LANG_SCNG(yy_leng));
187
        /* read the following character, either newline or ; */
188
0
        if (lex_scan(&token, NULL) != T_WHITESPACE) {
189
0
          zend_write((char*)LANG_SCNG(yy_text), LANG_SCNG(yy_leng));
190
0
        }
191
0
        zend_write("\n", sizeof("\n") - 1);
192
0
        prev_space = 1;
193
0
        ZVAL_UNDEF(&token);
194
0
        continue;
195
196
0
      default:
197
0
        zend_write((char*)LANG_SCNG(yy_text), LANG_SCNG(yy_leng));
198
0
        break;
199
0
    }
200
201
0
    if (Z_TYPE(token) == IS_STRING) {
202
0
      switch (token_type) {
203
0
        case T_OPEN_TAG:
204
0
        case T_OPEN_TAG_WITH_ECHO:
205
0
        case T_CLOSE_TAG:
206
0
        case T_WHITESPACE:
207
0
        case T_COMMENT:
208
0
        case T_DOC_COMMENT:
209
0
          break;
210
211
0
        default:
212
0
          zval_ptr_dtor_str(&token);
213
0
          break;
214
0
      }
215
0
    }
216
0
    prev_space = 0;
217
0
    ZVAL_UNDEF(&token);
218
0
  }
219
220
  /* Discard parse errors thrown during tokenization */
221
0
  zend_clear_exception();
222
0
}