| 
									
										
											  
											
												chore: make yuzu REUSE compliant
[REUSE] is a specification that aims at making file copyright
information consistent, so that it can be both human and machine
readable. It basically requires that all files have a header containing
copyright and licensing information. When this isn't possible, like
when dealing with binary assets, generated files or embedded third-party
dependencies, it is permitted to insert copyright information in the
`.reuse/dep5` file.
Oh, and it also requires that all the licenses used in the project are
present in the `LICENSES` folder, that's why the diff is so huge.
This can be done automatically with `reuse download --all`.
The `reuse` tool also contains a handy subcommand that analyzes the
project and tells whether or not the project is (still) compliant,
`reuse lint`.
Following REUSE has a few advantages over the current approach:
- Copyright information is easy to access for users / downstream
- Files like `dist/license.md` do not need to exist anymore, as
  `.reuse/dep5` is used instead
- `reuse lint` makes it easy to ensure that copyright information of
  files like binary assets / images is always accurate and up to date
To add copyright information of files that didn't have it I looked up
who committed what and when, for each file. As yuzu contributors do not
have to sign a CLA or similar I couldn't assume that copyright ownership
was of the "yuzu Emulator Project", so I used the name and/or email of
the commit author instead.
[REUSE]: https://reuse.software
Follow-up to b2eb10382941bef0914f4a0a4685b9033440aa9f
											
										 
											2022-05-15 02:06:02 +02:00
										 |  |  | // SPDX-FileCopyrightText: 2015 Citra Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2015-08-19 17:00:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  | #include <array>
 | 
					
						
							|  |  |  | #include <cmath>
 | 
					
						
							| 
									
										
										
										
											2018-08-29 15:42:53 +02:00
										 |  |  | #include <QPainter>
 | 
					
						
							| 
									
										
										
										
											2018-01-11 20:33:56 -07:00
										 |  |  | #include "yuzu/util/util.h"
 | 
					
						
							| 
									
										
										
										
											2023-10-07 17:26:04 +02:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | #include <windows.h>
 | 
					
						
							|  |  |  | #include "common/fs/file.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-19 17:00:56 -03:00
										 |  |  | QFont GetMonospaceFont() { | 
					
						
							| 
									
										
										
										
											2019-05-20 14:44:13 -04:00
										 |  |  |     QFont font(QStringLiteral("monospace")); | 
					
						
							| 
									
										
										
										
											2015-08-19 17:00:56 -03:00
										 |  |  |     // Automatic fallback to a monospace font on on platforms without a font called "monospace"
 | 
					
						
							|  |  |  |     font.setStyleHint(QFont::Monospace); | 
					
						
							|  |  |  |     font.setFixedPitch(true); | 
					
						
							|  |  |  |     return font; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | QString ReadableByteSize(qulonglong size) { | 
					
						
							| 
									
										
										
										
											2019-05-20 14:44:13 -04:00
										 |  |  |     static constexpr std::array units{"B", "KiB", "MiB", "GiB", "TiB", "PiB"}; | 
					
						
							|  |  |  |     if (size == 0) { | 
					
						
							|  |  |  |         return QStringLiteral("0"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const int digit_groups = std::min(static_cast<int>(std::log10(size) / std::log10(1024)), | 
					
						
							|  |  |  |                                       static_cast<int>(units.size())); | 
					
						
							|  |  |  |     return QStringLiteral("%L1 %2") | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         .arg(size / std::pow(1024, digit_groups), 0, 'f', 1) | 
					
						
							| 
									
										
										
										
											2019-05-20 14:44:13 -04:00
										 |  |  |         .arg(QString::fromUtf8(units[digit_groups])); | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-29 15:42:53 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | QPixmap CreateCirclePixmapFromColor(const QColor& color) { | 
					
						
							|  |  |  |     QPixmap circle_pixmap(16, 16); | 
					
						
							|  |  |  |     circle_pixmap.fill(Qt::transparent); | 
					
						
							|  |  |  |     QPainter painter(&circle_pixmap); | 
					
						
							| 
									
										
										
										
											2018-09-17 05:57:23 -04:00
										 |  |  |     painter.setRenderHint(QPainter::Antialiasing); | 
					
						
							| 
									
										
										
										
											2018-08-29 15:42:53 +02:00
										 |  |  |     painter.setPen(color); | 
					
						
							|  |  |  |     painter.setBrush(color); | 
					
						
							| 
									
										
										
										
											2018-09-17 05:57:23 -04:00
										 |  |  |     painter.drawEllipse({circle_pixmap.width() / 2.0, circle_pixmap.height() / 2.0}, 7.0, 7.0); | 
					
						
							| 
									
										
										
										
											2018-08-29 15:42:53 +02:00
										 |  |  |     return circle_pixmap; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-10-07 17:26:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | bool SaveIconToFile(const std::string_view path, const QImage& image) { | 
					
						
							|  |  |  | #if defined(WIN32)
 | 
					
						
							|  |  |  | #pragma pack(push, 2)
 | 
					
						
							|  |  |  |     struct IconDir { | 
					
						
							|  |  |  |         WORD id_reserved; | 
					
						
							|  |  |  |         WORD id_type; | 
					
						
							|  |  |  |         WORD id_count; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct IconDirEntry { | 
					
						
							|  |  |  |         BYTE width; | 
					
						
							|  |  |  |         BYTE height; | 
					
						
							|  |  |  |         BYTE color_count; | 
					
						
							|  |  |  |         BYTE reserved; | 
					
						
							|  |  |  |         WORD planes; | 
					
						
							|  |  |  |         WORD bit_count; | 
					
						
							|  |  |  |         DWORD bytes_in_res; | 
					
						
							|  |  |  |         DWORD image_offset; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | #pragma pack(pop)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QImage source_image = image.convertToFormat(QImage::Format_RGB32); | 
					
						
							|  |  |  |     constexpr int bytes_per_pixel = 4; | 
					
						
							|  |  |  |     const int image_size = source_image.width() * source_image.height() * bytes_per_pixel; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     BITMAPINFOHEADER info_header{}; | 
					
						
							|  |  |  |     info_header.biSize = sizeof(BITMAPINFOHEADER), info_header.biWidth = source_image.width(), | 
					
						
							|  |  |  |     info_header.biHeight = source_image.height() * 2, info_header.biPlanes = 1, | 
					
						
							|  |  |  |     info_header.biBitCount = bytes_per_pixel * 8, info_header.biCompression = BI_RGB; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const IconDir icon_dir{.id_reserved = 0, .id_type = 1, .id_count = 1}; | 
					
						
							|  |  |  |     const IconDirEntry icon_entry{.width = static_cast<BYTE>(source_image.width()), | 
					
						
							|  |  |  |                                   .height = static_cast<BYTE>(source_image.height() * 2), | 
					
						
							|  |  |  |                                   .color_count = 0, | 
					
						
							|  |  |  |                                   .reserved = 0, | 
					
						
							|  |  |  |                                   .planes = 1, | 
					
						
							|  |  |  |                                   .bit_count = bytes_per_pixel * 8, | 
					
						
							|  |  |  |                                   .bytes_in_res = | 
					
						
							|  |  |  |                                       static_cast<DWORD>(sizeof(BITMAPINFOHEADER) + image_size), | 
					
						
							|  |  |  |                                   .image_offset = sizeof(IconDir) + sizeof(IconDirEntry)}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Common::FS::IOFile icon_file(path, Common::FS::FileAccessMode::Write, | 
					
						
							|  |  |  |                                  Common::FS::FileType::BinaryFile); | 
					
						
							|  |  |  |     if (!icon_file.IsOpen()) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!icon_file.Write(icon_dir)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!icon_file.Write(icon_entry)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!icon_file.Write(info_header)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int y = 0; y < image.height(); y++) { | 
					
						
							|  |  |  |         const auto* line = source_image.scanLine(source_image.height() - 1 - y); | 
					
						
							|  |  |  |         std::vector<u8> line_data(source_image.width() * bytes_per_pixel); | 
					
						
							|  |  |  |         std::memcpy(line_data.data(), line, line_data.size()); | 
					
						
							|  |  |  |         if (!icon_file.Write(line_data)) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     icon_file.Close(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } |