StapleGL
Header-only C++20 OpenGL wrapper
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Pages Concepts
shader.hpp
Go to the documentation of this file.
1
17#pragma once
18
19#include "gl_functions.hpp"
20#include "utility.hpp"
21#include <cstdint>
22#include <exception>
23#include <optional>
24#include <span>
25#include <string>
26#include <string_view>
27#include <unordered_map>
28#include <vector>
29
30namespace staplegl {
31
38enum class shader_type {
39 vertex,
44};
45
46inline std::string shader_type_to_string(shader_type type) noexcept;
47
57struct shader {
59 std::string source;
60};
61
78public:
79 shader_program() = default;
80
90 shader_program(std::string_view name, std::string_view path) noexcept;
91
101 shader_program(std::string_view name, std::initializer_list<std::pair<shader_type, std::string_view>> shaders) noexcept;
102
114 shader_program(std::string_view path) noexcept;
115
117 auto operator=(const shader_program&) -> shader_program& = default;
118
120 : m_shaders { std::move(other.m_shaders) }
121 , m_uniform_cache { std::move(other.m_uniform_cache) }
122 , m_id { other.m_id }
123 , m_name { std::move(other.m_name) }
124 {
125 other.m_id = 0;
126 }
127
128 auto operator=(shader_program&& other) noexcept -> shader_program&
129 {
130 if (this != &other) {
131 m_shaders = std::move(other.m_shaders);
132 m_uniform_cache = std::move(other.m_uniform_cache);
133 m_id = other.m_id;
134 m_name = std::move(other.m_name);
135 other.m_id = 0;
136 }
137 return *this;
138 }
139
145
146public:
151 void bind() const;
152
157 void unbind() const;
158
165 void upload_uniform1i(std::string_view name, int val);
166
173 void upload_uniform1f(std::string_view name, float val);
174
182 void upload_uniform2f(std::string_view name, float val0, float val1);
183
192 void upload_uniform3f(std::string_view name, float val0, float val1, float val2);
193
203 void upload_uniform4f(std::string_view name, float val0, float val1, float val2, float val3);
204
211 void upload_uniform_mat4f(std::string_view name, float const* mat);
212
219 void upload_uniform_mat3f(std::string_view name, float const* mat);
220
226 [[nodiscard]] auto constexpr program_id() const -> std::uint32_t;
227
233 [[nodiscard]] auto constexpr name() const -> std::string;
234
235public:
243 auto operator[](std::size_t index) -> shader&;
244
252 auto operator[](std::size_t index) const -> const shader&;
253
264 [[nodiscard]] static auto is_valid(std::uint32_t id) -> bool;
265
266private:
272 [[nodiscard]] auto create_program() const -> std::uint32_t;
273
282 [[nodiscard]] auto compile(shader_type shader_type, std::string_view source) const -> std::uint32_t;
283
297 [[nodiscard]] auto parse_shaders(std::string_view source) const -> std::vector<shader>;
298
299private:
306 [[nodiscard]] auto uniform_location(std::string_view name) -> int;
307
308private:
316 static constexpr auto to_gl_type(shader_type shader_type) -> std::uint32_t;
317
325 [[nodiscard]] static auto string_to_shader_type(std::string_view str) -> std::optional<shader_type>;
326
327private:
328 std::vector<shader> m_shaders;
329 std::unordered_map<std::string_view, int> m_uniform_cache;
330 std::uint32_t m_id {};
331 std::string m_name;
332};
333
334/*
335
336 IMPLEMENTATIONS
337
338*/
339
340inline shader_program::shader_program(std::string_view name, std::string_view path) noexcept
341 : m_shaders { parse_shaders(util::read_file(path)) }
342 , m_uniform_cache {}
343 , m_id(create_program())
344 , m_name { name }
345{
346}
347
348inline shader_program::shader_program(std::string_view name,
349 std::initializer_list<std::pair<shader_type, std::string_view>> shaders) noexcept
350 : m_id(create_program())
351 , m_name { name }
352{
353 for (const auto& [type, path] : shaders)
354 m_shaders.push_back({ type, util::read_file(path) });
355}
356
357inline shader_program::shader_program(std::string_view path) noexcept
358 : shader_program { util::get_file_name(path), path }
359{
360}
361
363{
364 glDeleteProgram(m_id);
365}
366
367inline void shader_program::bind() const
368{
369 glUseProgram(m_id);
370}
371
372inline void shader_program::unbind() const
373{
374 glUseProgram(0);
375}
376
377inline void shader_program::upload_uniform1i(std::string_view name, int val)
378{
379 glUniform1i(uniform_location(name), val);
380}
381
382inline void shader_program::upload_uniform1f(std::string_view name, float val)
383{
384 glUniform1f(uniform_location(name), val);
385}
386
387inline void shader_program::upload_uniform2f(std::string_view name, float val0, float val1)
388{
389 glUniform2f(uniform_location(name), val0, val1);
390}
391
392inline void shader_program::upload_uniform3f(std::string_view name, float val0, float val1, float val2)
393{
394 glUniform3f(uniform_location(name), val0, val1, val2);
395}
396
397inline void shader_program::upload_uniform4f(std::string_view name, float val0, float val1, float val2, float val3)
398{
399 glUniform4f(uniform_location(name), val0, val1, val2, val3);
400}
401
402inline void shader_program::upload_uniform_mat4f(std::string_view name, float const* mat)
403{
404 glUniformMatrix4fv(uniform_location(name), 1, GL_FALSE, mat);
405}
406
407inline void shader_program::upload_uniform_mat3f(std::string_view name, float const* mat)
408{
409 glUniformMatrix3fv(uniform_location(name), 1, GL_FALSE, mat);
410}
411
412inline constexpr auto shader_program::program_id() const -> std::uint32_t
413{
414 return m_id;
415}
416
417inline constexpr auto shader_program::name() const -> std::string
418{
419 return m_name;
420}
421
422inline auto shader_program::operator[](std::size_t index) -> shader&
423{
424 return m_shaders[index];
425}
426
427inline auto shader_program::operator[](std::size_t index) const -> const shader&
428{
429 return m_shaders[index];
430}
431
432inline auto shader_program::create_program() const -> std::uint32_t
433{
434 const std::uint32_t program { glCreateProgram() };
435 std::vector<std::uint32_t> shader_ids;
436
437 shader_ids.reserve(m_shaders.size());
438 for (const auto& [type, src] : m_shaders) {
439 shader_ids.push_back(compile(type, src));
440 }
441
442 for (const auto& id : shader_ids) {
443 glAttachShader(program, id);
444 }
445 glLinkProgram(program);
446
447 int link_success = 0;
448 glGetProgramiv(program, GL_LINK_STATUS, &link_success);
449
450 if (!link_success) [[unlikely]] {
451#ifdef STAPLEGL_DEBUG
452 int max_length {};
453 glGetProgramiv(program, GL_INFO_LOG_LENGTH, &max_length);
454 std::vector<char> error_log(max_length);
455 glGetProgramInfoLog(program, max_length, &max_length, &error_log[0]);
456 std::fprintf(stderr, STAPLEGL_LINEINFO ", failed to link shader program: %s\n",
457 error_log.data());
458#endif // STAPLEGL_DEBUG
459 glDeleteProgram(program);
460 return 0;
461 }
462
463 // Detach and delete shaders after linking the program.
464 for (const auto& id : shader_ids) {
465 glDetachShader(program, id);
466 glDeleteShader(id);
467 }
468
469 glValidateProgram(program);
470
471 int success = 0;
472 glGetProgramiv(program, GL_VALIDATE_STATUS, &success);
473 if (!success) [[unlikely]] {
474#ifdef STAPLEGL_DEBUG
475 int max_length {};
476 glGetProgramiv(program, GL_INFO_LOG_LENGTH, &max_length);
477 std::string error_log(max_length, '\0');
478 glGetProgramInfoLog(program, max_length, &max_length, &error_log[0]);
479 std::fprintf(stderr, STAPLEGL_LINEINFO ", failed to validate shader program: %s\n",
480 error_log.data());
481#endif // STAPLEGL_DEBUG
482 glDeleteProgram(program);
483 return 0;
484 }
485
486 return program;
487}
488
489inline auto shader_program::compile(shader_type shader_type, std::string_view source) const -> std::uint32_t
490{
491 const std::uint32_t id { glCreateShader(to_gl_type(shader_type)) };
492
493 // taking a pointer to this temporary allows us to get its address, without this intermediate
494 // variable we are forced to use the address of the temporary, which is not allowed.
495 // if anyone knows a better way to do this, please let me know.
496 const char* src { source.data() };
497
498 glShaderSource(id, 1, &src, nullptr);
499 glCompileShader(id);
500
501 int comp_ok {};
502 glGetShaderiv(id, GL_COMPILE_STATUS, &comp_ok);
503
504 if (comp_ok == GL_FALSE) [[unlikely]] {
505#ifdef STAPLEGL_DEBUG
506 int max_length {};
507 glGetShaderiv(id, GL_INFO_LOG_LENGTH, &max_length);
508 std::string error_log(max_length, '\0');
509 glGetShaderInfoLog(id, max_length, &max_length, &error_log[0]);
510 int32_t type_id {};
511 glGetShaderiv(id, GL_SHADER_TYPE, &type_id);
512 std::string shader_type_str = shader_type_to_string(static_cast<staplegl::shader_type>(type_id));
513 std::fprintf(stderr, STAPLEGL_LINEINFO ", failed to compile %s shader: \n%s\n",
514 shader_type_str.data(), error_log.data());
515#endif // STAPLEGL_DEBUG
516 glDeleteShader(id);
517 return 0;
518 }
519
520 return id;
521}
522
523inline auto shader_program::parse_shaders(std::string_view source) const -> std::vector<shader>
524{
525 std::vector<shader> shaders;
526 std::string_view const type_token { "#type" };
527
528 std::size_t pos { source.find(type_token, 0) };
529 while (pos != std::string::npos) {
530 const std::size_t eol { source.find_first_of("\r\n", pos) };
531 const std::size_t begin { pos + type_token.size() + 1 };
532 const std::size_t next_line_pos { source.find_first_not_of("\r\n", eol) };
533 std::string const type { source.substr(begin, eol - begin) };
534
535 auto const shader_type { string_to_shader_type(type) };
536
537 // With C++23 we could drastically simplify this thanks to std::optional's monadic operations.
538 if (!shader_type.has_value()) [[unlikely]] {
539
540#ifdef STAPLEGL_DEBUG
541 std::fprintf(stderr, STAPLEGL_LINEINFO ", invalid shader type \"%s\"\n",
542 type.data());
543#endif // STAPLEGL_DEBUG
544 return {};
545 }
546
547 pos = source.find(type_token, next_line_pos);
548 shaders.push_back({ shader_type.value(),
549 (pos == std::string::npos)
550 ? std::string(source.substr(next_line_pos))
551 : std::string(source.substr(next_line_pos, pos - next_line_pos)) });
552 }
553
554 return shaders;
555}
556
557inline auto shader_program::uniform_location(std::string_view name) -> int
558{
559 if (m_uniform_cache.find(name) != m_uniform_cache.end()) [[likely]] {
560 return m_uniform_cache[name];
561 } else {
562 const int location { glGetUniformLocation(m_id, name.data()) };
563#ifdef STAPLEGL_DEBUG
564 if (location == -1) {
565 std::fprintf(stderr, STAPLEGL_LINEINFO ", uniform \"%s\" not found in shader program \"%s\"\n",
566 name.data(), m_name.data());
567 }
568#endif // STAPLEGL_DEBUG
569 m_uniform_cache[name] = location;
570 return location;
571 }
572}
573
574inline auto shader_program::is_valid(std::uint32_t id) -> bool
575{
576 int link_ok {};
577 glGetProgramiv(id, GL_LINK_STATUS, &link_ok);
578
579 if (link_ok == GL_FALSE) [[unlikely]] {
580
581#ifdef STAPLEGL_DEBUG
582 int max_length {};
583 glGetProgramiv(id, GL_INFO_LOG_LENGTH, &max_length);
584 std::string error_log(max_length, '\0');
585 glGetProgramInfoLog(id, max_length, &max_length, &error_log[0]);
586 std::fprintf(stderr, STAPLEGL_LINEINFO ", failed to link shader program: %s\n",
587 error_log.data());
588#endif // STAPLEGL_DEBUG
589
590 glDeleteProgram(id);
591 return false;
592 }
593
594 glValidateProgram(id);
595
596 int success = 0;
597 glGetProgramiv(id, GL_VALIDATE_STATUS, &success);
598 if (!success) [[unlikely]] {
599#ifdef STAPLEGL_DEBUG
600 int max_length {};
601 glGetProgramiv(id, GL_INFO_LOG_LENGTH, &max_length);
602 std::string error_log(max_length, '\0');
603 glGetProgramInfoLog(id, max_length, &max_length, &error_log[0]);
604 std::fprintf(stderr, STAPLEGL_LINEINFO ", failed to validate shader program: %s\n",
605 error_log.data());
606#endif // STAPLEGL_DEBUG
607 glDeleteProgram(id);
608 return false;
609 }
610
611 return true;
612}
613
614inline constexpr auto shader_program::to_gl_type(shader_type shader_type) -> std::uint32_t
615{
616 switch (shader_type) {
618 return GL_VERTEX_SHADER;
620 return GL_FRAGMENT_SHADER;
622 return GL_TESS_CONTROL_SHADER;
624 return GL_TESS_EVALUATION_SHADER;
626 return GL_GEOMETRY_SHADER;
627 default:
628#ifdef STAPLEGL_DEBUG
629 std::fprintf(stderr, STAPLEGL_LINEINFO ", invalid shader type enum %d, \n",
630 static_cast<int>(shader_type));
631#endif // STAPLEGL_DEBUG
632 std::terminate();
633 }
634}
635
636inline auto shader_program::string_to_shader_type(std::string_view str) -> std::optional<shader_type>
637{
638 static std::unordered_map<std::string_view, shader_type> map {
639 { "vertex", shader_type::vertex },
640 { "fragment", shader_type::fragment },
641 { "tess_control", shader_type::tess_control },
642 { "tess_eval", shader_type::tess_eval },
643 { "geometry", shader_type::geometry }
644 };
645
646 if (map.find(str) != map.end()) [[likely]] {
647 return map[str];
648 }
649
650 return std::nullopt;
651}
652}
653
655{
656 switch (type) {
657 case shader_type::vertex:
658 return "vertex";
659 case shader_type::fragment:
660 return "fragment";
661 case shader_type::tess_control:
662 return "tess_control";
663 case shader_type::tess_eval:
664 return "tess_eval";
665 case shader_type::geometry:
666 return "geometry";
667 default:
668 return "unknown";
669 }
670}
Shader program class.
Definition shader.hpp:77
void unbind() const
Unbind the shader program.
Definition shader.hpp:372
auto uniform_location(std::string_view name) -> int
Obtain the location of a uniform in the shader program.
Definition shader.hpp:557
auto constexpr name() const -> std::string
Obtain the shader program name.
Definition shader.hpp:417
void upload_uniform3f(std::string_view name, float val0, float val1, float val2)
Upload a 3D float uniform to the shader program.
Definition shader.hpp:392
static constexpr auto to_gl_type(shader_type shader_type) -> std::uint32_t
Convert a shader type to its OpenGL equivalent.
Definition shader.hpp:614
void upload_uniform2f(std::string_view name, float val0, float val1)
Upload a 2D float uniform to the shader program.
Definition shader.hpp:387
auto create_program() const -> std::uint32_t
Create a program object.
Definition shader.hpp:432
~shader_program()
Destroy the shader program object.
Definition shader.hpp:362
void bind() const
Bind the shader program.
Definition shader.hpp:367
void upload_uniform_mat3f(std::string_view name, float const *mat)
Upload a 3x3 float matrix uniform to the shader program.
Definition shader.hpp:407
std::unordered_map< std::string_view, int > m_uniform_cache
Definition shader.hpp:329
std::vector< shader > m_shaders
Definition shader.hpp:328
void upload_uniform1f(std::string_view name, float val)
Upload a float uniform to the shader program.
Definition shader.hpp:382
static auto string_to_shader_type(std::string_view str) -> std::optional< shader_type >
Convert a string to a shader type.
Definition shader.hpp:636
shader_program(const shader_program &)=default
auto compile(shader_type shader_type, std::string_view source) const -> std::uint32_t
Create a shader object.
Definition shader.hpp:489
auto parse_shaders(std::string_view source) const -> std::vector< shader >
Link the shader program.
Definition shader.hpp:523
auto operator[](std::size_t index) -> shader &
Obtain a reference to a shader in the shader program.
Definition shader.hpp:422
void upload_uniform4f(std::string_view name, float val0, float val1, float val2, float val3)
Upload a 4D float uniform to the shader program.
Definition shader.hpp:397
auto operator=(const shader_program &) -> shader_program &=default
static auto is_valid(std::uint32_t id) -> bool
Check if a shader program is valid.
Definition shader.hpp:574
void upload_uniform_mat4f(std::string_view name, float const *mat)
Upload a 4x4 float matrix uniform to the shader program.
Definition shader.hpp:402
shader_program(shader_program &&other) noexcept
Definition shader.hpp:119
auto operator=(shader_program &&other) noexcept -> shader_program &
Definition shader.hpp:128
void upload_uniform1i(std::string_view name, int val)
Upload an integer uniform to the shader program.
Definition shader.hpp:377
auto constexpr program_id() const -> std::uint32_t
Obtain the shader program id.
Definition shader.hpp:412
Loads OpenGL functions.
static auto get_file_name(std::string_view path) -> std::string
Get the filename without the extension.
Definition utility.hpp:81
static auto read_file(std::string_view path) -> std::string
Read a file into a string.
Definition utility.hpp:58
std::string shader_type_to_string(shader_type type) noexcept
Definition shader.hpp:654
shader_type
The type of the shader.
Definition shader.hpp:38
Individual shader struct.
Definition shader.hpp:57
shader_type type
Definition shader.hpp:58
std::string source
Definition shader.hpp:59
Utility functions for parsing files.