苹果系统UTM指定版本安装

只有一台MacMiniM1的系统,如何在虚拟机安装多个MacOS版本

在MacOS系统上,支持M1的虚拟化的有VMWare FussionParallels Desktop以及开源版本UTM

前者不支持在MacOS内部再虚拟化MacOS系统,PD没有安装验证。

UTM支持MacOS系统,但要求Mac12+系统的虚拟化。

在苹果官网上,不支持MacOS11以上的系统ipsw文件下载,线上国内也很少相关的下载。

国外专门有人提供了所有Mac的系统下载。https://ipsw.me/

因此我选择MacOS12.1的版本下载:
https://updates.cdn-apple.com/2021FCSWinter/fullrestores/002-42433/F3F6D5CD-67FE-449C-9212-F7409808B6C4/UniversalMac_12.1_21C52_Restore.ipsw
https://updates.cdn-apple.com/2022FallFCS/fullrestores/012-60270/0A7F49BA-FC31-4AD9-8E45-49B1FB9128A6/UniversalMac_13.1_22C65_Restore.ipsw

Qt6.2.4在多屏显示时存在重大BUG

这是Qt6.2.4版本的多屏代码,在同一品牌和同一型号时会存在BUG.
以下是Qt6.2.3的代码,经检查Qt5.15.2/Qt6.2.5/Qt6.8.2都不会存在该BUG

static bool monitorData(HMONITOR hMonitor, QWindowsScreenData *data)
{
    MONITORINFOEX info;
    memset(&info, 0, sizeof(MONITORINFOEX));
    info.cbSize = sizeof(MONITORINFOEX);
    if (GetMonitorInfo(hMonitor, &info) == FALSE)
        return false;

    data->hMonitor = hMonitor;
    data->geometry = QRect(QPoint(info.rcMonitor.left, info.rcMonitor.top), QPoint(info.rcMonitor.right - 1, info.rcMonitor.bottom - 1));
    data->availableGeometry = QRect(QPoint(info.rcWork.left, info.rcWork.top), QPoint(info.rcWork.right - 1, info.rcWork.bottom - 1));
》》》》》
  问题在这行开始,它通过getPathInfo()获取一个友名,然后如果是相同品牌且同一型号时,其友名是相同的。从而导致data->name被初始化为友名。
    DISPLAYCONFIG_PATH_INFO pathInfo = {};
    const bool hasPathInfo = getPathInfo(info, &pathInfo);
    if (hasPathInfo) {
        DISPLAYCONFIG_TARGET_DEVICE_NAME deviceName = {};
        deviceName.header.type = DISPLAYCONFIG_DEVICE_INFO_GET_TARGET_NAME;
        deviceName.header.size = sizeof(DISPLAYCONFIG_TARGET_DEVICE_NAME);
        deviceName.header.adapterId = pathInfo.targetInfo.adapterId;
        deviceName.header.id = pathInfo.targetInfo.id;
        if (DisplayConfigGetDeviceInfo(&deviceName.header) == ERROR_SUCCESS)
            data->name = QString::fromWCharArray(deviceName.monitorFriendlyDeviceName);
    }
    if (data->name.isEmpty())
        data->name = QString::fromWCharArray(info.szDevice);


《《《《《《
    if (wcscmp(info.szDevice, L"WinDisc") == 0) {
        data->flags |= QWindowsScreenData::LockScreen;
    } else {
        if (const HDC hdc = CreateDC(info.szDevice, nullptr, nullptr, nullptr)) {
            const QDpi dpi = monitorDPI(hMonitor);
            data->dpi = dpi.first > 0 ? dpi : deviceDPI(hdc);
            data->depth = GetDeviceCaps(hdc, BITSPIXEL);
            data->format = data->depth == 16 ? QImage::Format_RGB16 : QImage::Format_RGB32;
            data->physicalSizeMM = QSizeF(GetDeviceCaps(hdc, HORZSIZE), GetDeviceCaps(hdc, VERTSIZE));
            const int refreshRate = GetDeviceCaps(hdc, VREFRESH);
            if (refreshRate > 1) // 0,1 means hardware default.
                data->refreshRateHz = refreshRate;
            DeleteDC(hdc);
        } else {
            qWarning("%s: Unable to obtain handle for monitor '%s', defaulting to %g DPI.",
                     __FUNCTION__, qPrintable(QString::fromWCharArray(info.szDevice)),
                     data->dpi.first);
        } // CreateDC() failed
    } // not lock screen

以下是Qt6.2.3的代码,经检查Qt5.15.2/Qt6.2.5/Qt6.8.2都不会存在该BUG:

static bool monitorData(HMONITOR hMonitor, QWindowsScreenData *data)
{
    MONITORINFOEX info;
    memset(&info, 0, sizeof(MONITORINFOEX));
    info.cbSize = sizeof(MONITORINFOEX);
    if (GetMonitorInfo(hMonitor, &info) == FALSE)
        return false;

    data->hMonitor = hMonitor;
    data->geometry = QRect(QPoint(info.rcMonitor.left, info.rcMonitor.top), QPoint(info.rcMonitor.right - 1, info.rcMonitor.bottom - 1));
    data->availableGeometry = QRect(QPoint(info.rcWork.left, info.rcWork.top), QPoint(info.rcWork.right - 1, info.rcWork.bottom - 1));
    data->name = QString::fromWCharArray(info.szDevice);
    if (data->name == u"WinDisc") {
        data->flags |= QWindowsScreenData::LockScreen;
    } else {
        if (const HDC hdc = CreateDC(info.szDevice, nullptr, nullptr, nullptr)) {
            const QDpi dpi = monitorDPI(hMonitor);
            data->dpi = dpi.first > 0 ? dpi : deviceDPI(hdc);
            data->depth = GetDeviceCaps(hdc, BITSPIXEL);
            data->format = data->depth == 16 ? QImage::Format_RGB16 : QImage::Format_RGB32;
            data->physicalSizeMM = QSizeF(GetDeviceCaps(hdc, HORZSIZE), GetDeviceCaps(hdc, VERTSIZE));
            const int refreshRate = GetDeviceCaps(hdc, VREFRESH);
            if (refreshRate > 1) // 0,1 means hardware default.
                data->refreshRateHz = refreshRate;
            DeleteDC(hdc);
        } else {
            qWarning("%s: Unable to obtain handle for monitor '%s', defaulting to %g DPI.",
                     __FUNCTION__, qPrintable(QString::fromWCharArray(info.szDevice)),
                     data->dpi.first);
        } // CreateDC() failed
    } // not lock screen

安装VS2019

线上只有VS2022的安装下载,而VS2019(Visual studio 2019)版本已经被隐藏起来了,如何找出来呢?

https://learn.microsoft.com/zh-cn/visualstudio/releases/2019/history

在fedora30下安装Qt6.2.4遇到的问题

一、安装编译环境

sudo dnf groupinstall "development tools"
sudo dnf install g++ libxkbcommon-devel libxkbcommon-x11-devel

二、安装Qt6.2.4

./qt-unified-linux-x64-online.run --mirror http://mirrors.ustc.edu.cn/qtproject/
若无法启动,可能需要安装xcb以下:
sudo dnf install xcb-util-cursor xcb-util-cursor-devel 

三、配置cmake环境变量

export PATH=/home/abc/Qt/Tools/CMake/bin:$PATH

PNG批量转换成SVG图

1.创建png2svg.ps1脚本

# 临时添加 Inkscape 的路径到当前会话的 PATH
$env:Path += ";H:\tools\png2svg\ImageMagick-7.1.1;H:\tools\png2svg\potrace-1.16.win64"

# inkscape --action-list
# pause

# 设置输入目录(包含 PNG 文件的目录)
$input_dir = "D:\aoyiduo\woterm\woterm\private\skins\blue"
#$input_dir = "H:\tools\png2svg\skin"

# 设置输出目录(与输入目录平级的 svg 目录)
$output_dir = Join-Path -Path (Split-Path -Path $input_dir -Parent) -ChildPath "svg"

# 如果输出目录不存在,则创建
if (-not (Test-Path $output_dir)) {
    New-Item -ItemType Directory -Path $output_dir | Out-Null
}

# 获取目录中的所有 PNG 文件
$png_files = Get-ChildItem -Path $input_dir -Filter *.png

# 遍历所有 PNG 文件
foreach ($file in $png_files) {
    try {
        # 设置输出文件路径(在输出目录中生成对应文件名的 SVG 文件)
        $temp_file = Join-Path -Path $output_dir -ChildPath ([System.IO.Path]::ChangeExtension($file.Name, ".bmp"))
        $output_file = Join-Path -Path $output_dir -ChildPath ([System.IO.Path]::ChangeExtension($file.Name, ".svg"))
        
        # 如果目标文件已存在,先删除旧文件
        if (Test-Path $output_file) {
            Write-Output "File already exists, deleting: $($output_file)"
            Remove-Item -Path $output_file -Force
        }
        # alpha default on
        # magick $file.FullName -alpha on $temp_file
        magick $file.FullName $temp_file
        # 背景是黑色,图案是透明色。
        # potrace $temp_file -s -o $output_file
        # 图案是黑色,背景是透明色。
        potrace $temp_file --invert -s -o $output_file
        
        # pause
        Remove-Item $temp_file

        # 打印成功信息
        Write-Output "Converted to vector: $($file.Name) -> $($output_file)"
    } catch {
        # 打印错误信息
        Write-Error "Failed to convert: $($file.FullName). Error: $_"
    }
}

2.创建一个调用脚本,用于临时解决脚本权限问题

 powershell -ExecutionPolicy Bypass -File "png2svg.ps1"
 pause

在openwrt系统中,其ssh是dropbear开发的简化版

在openwrt系统中,其ssh是dropbear开发的简化版,故它是不能直接openssh产生的公私钥。需要使用专用的dropbearkey命令来生成私钥及提取公钥。

dropbearkey -t rsa -f ~/.ssh/id_dropbear
dropbearkey -y -f ~/.ssh/id_dropbear
rsync -avz -e "ssh -i /path/to/private_key" /local/path/ user@remote_server_ip:/remote/path/

如何让VS编码体验如同QtCreator一样

已经习惯Qt开发后,转到VS2022后,VC的一些休验是很不适应的。

第一步:安装Qt-Assist和QtClasss向导插件【扩展》管理扩展】。

第二步:修改编码风格【工具》选项》文本编辑器》C/C++》代码样式》格式设置】
》保持原有的VisualSudio设置样式。
》换行》不要对单行代码块换行。
》新行》大括号的一些设置。

第三步:修改快捷键【工具》选项》环境》键盘】选择VisualAssist(VisualStudio)的键盘方案。

openwrt的rsync库手动安装

最近折腾openwrt相关的服务器工具,发现rsync直接用opkg命令无法安装rsync至系统中,故可采用手动方安装。

第一步:找到openwrt的仓库。
https://downloads.openwrt.org/releases/18.06.9/packages/aarch64_cortex-a53/

第二步:openwrt相关子库有base/luci/routing/telephony等。在这几个子库目录下搜索rsync,然后在https://downloads.openwrt.org/releases/18.06.9/packages/aarch64_cortex-a53/packages目录下,找到路径:https://downloads.openwrt.org/releases/18.06.9/packages/aarch64_cortex-a53/packages/rsync_3.1.3-1_aarch64_cortex-a53.ipk

第三步:本地化安装:
opkg install ./rsync_3.1.3-1_aarch64_cortex-a53.ipk

openwrt的后台服务

第一步骤:service,将显示所以服务。

第二步:找一个合适的服务,如sshd

第三步:进行/etc/init.d/路径。

第四步:上述步骤均不成功时,可执行下述脚本。

#!/bin/bash

path_current=`pwd`
path_script=$(cd "$(dirname "$0")"; pwd)
mode=$1

app_process=`ps| grep "frpc"| grep -v grep`

case "$mode" in
   'start')
        echo "it's ready to start op...."
        if test -n "$app_process"; then
                echo ""
                echo "$app_process"
                echo ""
        else
                cd $path_script   #进入脚本所在目录下,目的是使springboot的config目录生效。
                ${path_script}/frpc -c ${path_script}/frpc.ini > /dev/null 2>&1 &
                cd $path_current

        fi

        echo 'success to start.'
        ;;
   'stop')
        echo "it's ready to check process..."
        if test -n "$app_process"; then
                echo "had find app process informaton"
                echo $app_process | awk '{print ($1)}' | xargs kill -9
        fi
        echo 'success to kill.'
        ;;
    *)
        basename=`basename "$0"`
        echo "Usage: $basename  {start|stop}  [ server options ]"
        exit 1
        ;;
esac
exit 1

第5步:加入计划任务

crontab -e

为zidoo-x9s安装nas服务

每次固件安装,均需要重启系统【系统》重启】,这重启动方式最彻底。

1.下载固件

原先系统为v2.1.28,从zidoo.tv官网下载v2.1.45的最新固件包,并存在百度云盘。

通过网盘分享的文件:zidoo-x9s固件
链接: https://pan.baidu.com/s/1CBj2I2ggl0h_5Dt4wM2nSA?pwd=2tgi 提取码: 2tgi

2.安装固件

第一步:升级至最新固件v2.1.45,以及最新的oenwrta-18.06的openwrt固件包。

第二步:安装RealtekAdbdAuto-v1.1.0-signed.apk和RealtekOpenwrtDNS-v1.0.0-signed.apk,前者是adbd的调试开启,后者是DNS的执行,若不开启是无法安装执行路由解析的如ping baidu.com命令是无法完成百度地址的转换。

安装包完后,重启系统。

3.静态化IP地址如192.168.0.110。

4.访问http://192.168.0.110网页,不用设置密钥,以防忘记。

a).添加公钥,采用证书方式访问。【系统 》管理权 》SSH访问】。
b).添加软件源。【系统 》管理权 》软件包》配置》自动义软件源】添加如下:

src/gz openwrt_core https://mirrors.cqupt.edu.cn/openwrt/releases/23.05.3/targets/YOUR_TARGET/aarch64_cortex-a53/packages
src/gz openwrt_base https://mirrors.cqupt.edu.cn/openwrt/releases/23.05.3/packages/aarch64_cortex-a53/base
src/gz openwrt_luci https://mirrors.cqupt.edu.cn/openwrt/releases/23.05.3/packages/aarch64_cortex-a53/luci
src/gz openwrt_packages https://mirrors.cqupt.edu.cn/openwrt/releases/23.05.3/packages/aarch64_cortex-a53/packages
src/gz openwrt_routing https://mirrors.cqupt.edu.cn/openwrt/releases/23.05.3/packages/aarch64_cortex-a53/routing
src/gz openwrt_telephony https://mirrors.cqupt.edu.cn/openwrt/releases/23.05.3/packages/aarch64_cortex-a53/telephony

5.挂载硬盘,务必确认硬盘是否挂载成功,设备地址是以/dev/xxx,但默认挂载地址是以/mnt/xxx形式,故输入地址时,请检查正确。

6.设置挂载盘随系统启动

方法一:添加启动项。【系统 》启动项 》 本地启动脚本】添加

mount -t vfat /dev/sda1 /mnt/sda1

方法二:编写挂载脚本:假如路径为/data/mysda1.sh,其内容如下:这方法行不通

uci add fstab mount
uci set fstab.@mount[-1].device='/dev/sda1'
uci set fstab.@mount[-1].target='/mnt/sda1'
uci set fstab.@mount[-1].fstype='vfat'
uci set fstab.@mount[-1].enabled='1'
uci commit fstab

检验/etc/config/fstab文件内容,已经在尾部添加如下代码:

config mount
        
       option device '/dev/sda1'
       option target '/mnt/sda1'
        
       option fstype 'vfat'
        
       option enabled '1'

6.开启nas服务。
在其模板中添加server min protocol = NT1,确保最低访问协议是smb1,这是小米的必须条件。

7.测试服务。