Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

macOS Sonoma 14.4 broke macchanger, and ifconfig!? wtfh #2

Open
artkiver opened this issue Mar 12, 2024 · 44 comments
Open

macOS Sonoma 14.4 broke macchanger, and ifconfig!? wtfh #2

artkiver opened this issue Mar 12, 2024 · 44 comments

Comments

@artkiver
Copy link

I already submitted Feedback to Apple, but figured I would report things here too in case someone may discover a workaround.

macchanger worked without issues on macOS 14.3.1.

After updating to 14.4:

% sudo macchanger -r en0                         
Password:
INFO:    Type of interface is Wi-Fi. Will disassociate from any network.
WARNING: The airport command line tool is deprecated and will be removed in a future release.
For diagnosing Wi-Fi related issues, use the Wireless Diagnostics app or wdutil command line tool.
ERROR:   This device / interface has no MAC address to set.

Worse, this doesn't seem to be unique to macchanger; ifconfig seems broken on macOS 14.4 Sonoma as well:

sudo ifconfig en0 ether 10:93:e8:37:38:39
ifconfig: ioctl (SIOCAIFADDR): Can't assign requested address
adomino@MBAm2starhikari ~ % sudo ifconfig en0 ether de:ad:be:ef:de:ad
ifconfig: ioctl (SIOCAIFADDR): Can't assign requested address

Tested as broken on two Apple Silicon devices which previously functioned OK with macchanger on 14.3.1.

@shilch
Copy link
Owner

shilch commented Mar 26, 2024

Thanks for reporting this issue! I just upgraded to Sonoma 14.4.1 and can confirm the issue.
This appears to be a problem with the airport command line tool as it doesn't disassociate from the network. You can manually disconnect from the Wi-Fi network and then run macchanger which works.

@shilch shilch pinned this issue Mar 26, 2024
@shilch
Copy link
Owner

shilch commented Mar 26, 2024

I tried to find an alternative to the airport command and came up with the following snippet that imitates IO80211.framework. Some information can also be found here.

It would be great if you could test if this code correctly performs disassociation from Wi-Fi. If you have access to older versions of macOS it would also be great to have a positive sign that they are working there.

The below code can be compiled using clang -o disassoc ./disassoc.c.

#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include <sys/ioctl.h>
#include <sys/socket.h>

int main(int argc, char** argv) {
    int fd = socket(AF_INET, SOCK_DGRAM, 0);

    struct io80211_ioctl {
        char     ifname[16];
        uint64_t type;
        uint32_t length;
        void*    data;
    } ioc;
    strcpy(ioc.ifname, "en0");
    ioc.type = 22;
    ioc.length = 0;
    ioc.data = NULL;

    int res = ioctl(fd, 0x802869c8, &ioc);
    if(res == -1) {
        perror("ioctl");
        return res;
    }
    return 0;
}

@DarkNero69
Copy link

Genius, how did you even pull this? Works now !

@shilch
Copy link
Owner

shilch commented Apr 4, 2024

Genius, how did you even pull this? Works now !

Great to hear that, @DarkNero69. Do you have the possibility to test the code on an older system?

@DarkNero69
Copy link

Genius, how did you even pull this? Works now !

Great to hear that, @DarkNero69. Do you have the possibility to test the code on an older system?

Hey, sorry for the delayed reply. Unfortunately i don't have another computer or any other system. I only got macbook air m1 :/ but this can easily be tested in public computers, though i don't have the time to hit up library at the moment haha

@aaronleetw
Copy link

@shilch The fix is tested to be working on my MBP 13-inch, 2020 running i5.
Be sure to run it with sudo though.

@shilch
Copy link
Owner

shilch commented Apr 9, 2024

Thanks @aaronleetw, @DarkNero69!

I developed a new version that supports macOS Sonoma 14.4+ on branch v0.2-draft.

Please let me know if the new version does the job:

git clone -b v0.2-draft https://github.com/shilch/macchanger
cd macchanger
make
./macchanger

@aaronleetw
Copy link

@shilch Works on my end! Thank you for your work.

@DarkNero69
Copy link

@shilch It works for me too! I'm not very experienced with the terminal, but I managed to get it working hehe. Amazing work, thank you! It's much simpler and faster to use now =)

@artkiver
Copy link
Author

Thanks @aaronleetw, @DarkNero69!

I developed a new version that supports macOS Sonoma 14.4+ on branch v0.2-draft.

Please let me know if the new version does the job:

git clone -b v0.2-draft https://github.com/shilch/macchanger
cd macchanger
make
./macchanger

Thank you!

I've tested the new v0.2-draft branch on 14.3.1, 14.4 and 14.4.1 and it is functioning OK.

Attempting to build on OS X El Capitan 10.11.6 yields the following errors:

make macchanger.c:39:51: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? void interface_get_ether(const interface_t iface, ether_addr_t* ether); ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:40:51: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? void interface_set_ether(interface_t iface, const ether_addr_t* ether); ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:41:61: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? void interface_get_permanent_ether(const interface_t iface, ether_addr_t* eth... ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:48:42: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? void change_mac(interface_t iface, const ether_addr_t* ether); ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:50:19: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? void random_ether(ether_addr_t* ether); ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:51:34: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? int ether_parse(const char* str, ether_addr_t* ether); ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:52:35: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? const char* ether_to_string(const ether_addr_t* ether); ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:65:5: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? ether_addr_t ether; ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:155:5: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? ether_addr_t ether, permanent_ether; ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:164:48: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? void change_mac(const interface_t iface, const ether_addr_t* new_ether) { ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:165:5: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? ether_addr_t permanent_ether, old_ether, current_ether; ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:167:18: error: member reference base type 'const user_addr_t' (aka 'const unsigned long long') is not a structure or union if (new_ether->octet[0] & 1) { ~~~~~~~~~^ ~~~~~ macchanger.c:194:19: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? void random_ether(ether_addr_t* ether) { ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:204:35: error: member reference base type 'user_addr_t' (aka 'unsigned long long') is not a structure or union ssize_t n = read(fd, ether->octet + (ETHER_ADDR_LEN - to_read), ... ~~~~~^ ~~~~~ macchanger.c:213:10: error: member reference base type 'user_addr_t' (aka 'unsigned long long') is not a structure or union ether->octet[0] &= 0xFE; ~~~~~^ ~~~~~ macchanger.c:219:34: error: unknown type name 'ether_addr_t'; did you mean 'user_addr_t'? int ether_parse(const char* str, ether_addr_t* ether) { ^~~~~~~~~~~~ user_addr_t /usr/include/i386/types.h:97:20: note: 'user_addr_t' declared here typedef u_int64_t user_addr_t; ^ macchanger.c:226:14: error: member reference base type 'user_addr_t' (aka 'unsigned long long') is not a structure or union ether->octet[i] = 0; ~~~~~^ ~~~~~ macchanger.c:228:18: error: member reference base type 'user_addr_t' (aka 'unsigned long long') is not a structure or union ether->octet[i] <<= 4; ~~~~~^ ~~~~~ macchanger.c:230:45: error: member reference base type 'user_addr_t' (aka 'unsigned long long') is not a structure or union if(ch >= '0' && ch <= '9') ether->octet[i] |= (ch - '0'); ~~~~~^ ~~~~~ fatal error: too many errors emitted, stopping now [-ferror-limit=] 20 errors generated. make: *** [macchanger] Error 1

However, the previous macchanger.sh functions OK on El Capitan 10.11.6.

@artkiver
Copy link
Author

I also noticed that on macOS versions with the v0.2-draft branch, when invoked without root privileges, it will fail to change the MAC address (expected behavior) but it will still disassociate from WiFi (I am not sure if this is desirable?).

@geek4what
Copy link

@shilch Hi there, the branch v0.2-draft didn't work on my macOS(Sonoma 14.4.1).

sudo ./macchanger -r en0
INFO:    Type of interface is Wi-Fi. Will disassociate from any network.
interface_set_ether: Can't assign requested address

@nilssonalex
Copy link

Thank you for this. It works with the wifi adapter on my M2 Macbook Air, Sonoma 14.5. But, it does not work with my Belkin USB-C ethernet adapter, i get the error message "interface_set_ether: Can't assign requested address". Any leads?

@DarkNero69
Copy link

Hello. Sorry i haven't been replied to anything yet. First of all, thank you so much for developing such an amazing thing!

I haven't changed my mac address lately, but i just tested it by typing "sudo macchanger" to terminal. It seems like it works, managed to change to some random mac address. Thanks!

@shilch
Copy link
Owner

shilch commented May 28, 2024

Sorry for not responding in a while.

@artkiver The compilation error occurs because the ether_addr_t did not exist in El Capitan. That should be an easy fix which I will add.

@geek4what Which device are you using? Is the en0 an Airport device?

@nilssonalex The tool uses the interface APIs provided by the operating system. If the network driver for this 3rd party device does not implement the possibility to change the mac address, there's nothing that macchanger can do. You would need a driver which has this feature.

@fharper
Copy link

fharper commented May 28, 2024

The v0.2-draft is also working for me, thanks!

@geek4what
Copy link

Sorry for not responding in a while.

@geek4what Which device are you using? Is the en0 an Airport device?

wifi en0.
By the way, after upgrade to 14.5, it's work like a charm.

@johndekroon
Copy link

For wifi this works great. However, if I try to change the mac address of a wired network, I still get the following error:
sh-3.2# ./macchanger -m [mac] en17
interface_set_ether: Can't assign requested address

I use 14.5. The interface is listed as a USB 10/100/1000 device.

@dennis777
Copy link

dennis777 commented Jun 12, 2024

@shilch Hi there, the branch v0.2-draft didn't work on my macOS(Sonoma 14.4.1).

sudo ./macchanger -r en0
INFO:    Type of interface is Wi-Fi. Will disassociate from any network.
interface_set_ether: Can't assign requested address

It's because macchanger can't disassociate from the wifi network. Disconnect from wifi yourself, then run macchanger and it will work. Tested on M1 Max MBP running macOS 15 Sequoia beta

@dennis777
Copy link

I tried to find an alternative to the airport command and came up with the following snippet that imitates IO80211.framework. Some information can also be found here.

It would be great if you could test if this code correctly performs disassociation from Wi-Fi. If you have access to older versions of macOS it would also be great to have a positive sign that they are working there.

The below code can be compiled using clang -o disassoc ./disassoc.c.

#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include <sys/ioctl.h>
#include <sys/socket.h>

int main(int argc, char** argv) {
    int fd = socket(AF_INET, SOCK_DGRAM, 0);

    struct io80211_ioctl {
        char     ifname[16];
        uint64_t type;
        uint32_t length;
        void*    data;
    } ioc;
    strcpy(ioc.ifname, "en0");
    ioc.type = 22;
    ioc.length = 0;
    ioc.data = NULL;

    int res = ioctl(fd, 0x802869c8, &ioc);
    if(res == -1) {
        perror("ioctl");
        return res;
    }
    return 0;
}

Sadly this didn't work for me in macOS 15 beta (I never tried on macOS 14)

@ASolod82
Copy link

v0.2-draft, Macbook Pro 16 2021 M1 Pro

works fine for wifi en0, but for en4

sudo ./macchanger -r en4
interface_set_ether: Can't assign requested address

USB 10/100/1000 LAN:

Bus: USB
Vendor Name: Realtek
Product Name: USB 10/100/1000 LAN
Vendor ID: 0x0bda
Product ID: 0x8153
USB Link Speed: Up to 5 Gb/s
Driver: com.apple.DriverKit.AppleUserECM
BSD Device Name: en4
MAC Address: 00:e0:4c:cc:3f:7d
AVB Support: No

@dennis777
Copy link

Just so I can understand correctly, with nothing plugged into your Mac (no hardwire internet connections, or network adapters), you can run sudo./macchanger -r en0 while the MacBook is connected to a wireless network? Meaning it disassociates, changes the MAC address, and then reconnects?

Thanks

@ASolod82
Copy link

ASolod82 commented Jun 12, 2024

Just so I can understand correctly, with nothing plugged into your Mac (no hardwire internet connections, or network adapters), you can run sudo./macchanger -r en0 while the MacBook is connected to a wireless network? Meaning it disassociates, changes the MAC address, and then reconnects?

Thanks

image

Meaning, I just reconnected my wifi using new mac address. But it doesn't work for my type-c ethernet adapter :(
image

@dennis777
Copy link

Were you connected to wifi on en0 before you ran the command? And what does it show you when you run en4?

@shilch
Copy link
Owner

shilch commented Jul 5, 2024

@johndekroon @dennis777 @ASolod82 Thanks for your tests!

Can you please check if it works / there are errors for the following cases:

  • Using the v0.2-draft version
  • Using the master version but disassociating using this snippet
  • Using the master version but disassociating using the macOS UI

@dennis777
Copy link

Certainly! Here are the results tested on my 21 MacBook Pro M1 Max macOS 15.0 beta (shouldn't be any different for 14 as airport is deprecated there as well)

Using the v0.2-draft version

Result: Does not disassociate from wifi, thus not changing the Mac address
Screenshot 2024-07-08 at 10 12 14
Manually disassociating, works just fine
image

Using the master version but disassociating using this snippet

Result: does not disassociate but prints a message

ioctl: Operation not supported on socket

So then as expected, the macchanger doesn't work because it doesn't disassociate using that snippet
image

seems like ioctl() call isn't working in disassoc.c

Using the master version but disassociating using the macOS UI

Result: Works perfectly fine
image

Hope this helps! Let me know if theres anything else I can test @shilch

@shilch
Copy link
Owner

shilch commented Jul 8, 2024

Thanks @dennis777, I guess that Apple changed the interface to disassociate WiFi in one of the recent versions ("Operation not supported on socket" hints at an API change).
I am currently using 14.4.1 on M1 Pro and v0.2-draft works for me. User @geek4what had issues on 14.4.1 but upgrading to 14.5 fixed it, apparently. I assume that issue affects macOS 15.0+.
The solution could be to implement the disassociation using IOKit instead of ioctl. The headers have been leaked here. Once I'm less busy with university, I'll have a look at it.

@dennis777
Copy link

Ah okay I didn't know about this tool until after I updated to the beta.

I will play around with those dumped headers and see if I can come up with anything in the meantime. But my experience with C is extremely limited. I wish I understood it better 😅 Thank you

@ldoench
Copy link

ldoench commented Aug 11, 2024

The v0.2-draft version worked fine. I am on 14.5 (Macbook Air)

Is it possible to make it installable by brew?

Also the git clone https://github.com/shilch/macchanger command in the readme does not clone the v0.2-draft.. You should adjust it to git clone -b v0.2-draft https://github.com/shilch/macchanger

@dennis777
Copy link

The v0.2-draft version worked fine.

Which model of Mac do you use and what version of macOS?

@kecinzer
Copy link

kecinzer commented Sep 6, 2024

v0.2-draft not working for me. I have macOS 14.6.1 (23G93) and trying to change mac for my ehternet en6.

interface_set_ether: Can't assign requested address

@insert-zkm
Copy link

@kecinzer ive macos 14.6.1 For me this is worked perfectly. Shortly in the system settings create virtual interface for the actual interface and you could easily change mac address of virtual interface with ifconfig

@artkiver
Copy link
Author

Still seemed to be functioning OK for me with 14.7.

However with macOS 15.0, it seems as if WiFi must be manually turned off, then back on, otherwise this error is encountered:

macchanger -r en0 INFO: Type of interface is Wi-Fi. Will disassociate from any network. interface_set_ether: Can't assign requested address

My guess is something changed with macOS Sequoia that broke the recent disassoc.c iteration?

@ncihnegn
Copy link

ncihnegn commented Jan 3, 2025

Need to manually disassociate to work on 15.2.

@shilch
Copy link
Owner

shilch commented Jan 3, 2025

Hi everybody, sorry for the silence. I found a solution that automatically disassociates the wifi on newer macOS versions. Successfully tested on 15.2. I pushed the draft to a new branch v0.2-draft2.

Please let me know if this solution works for you and which version of macOS you are using.

git clone -b v0.2-draft2 https://github.com/shilch/macchanger
cd macchanger
make
sudo ./macchanger -r en0

@shilch
Copy link
Owner

shilch commented Jan 3, 2025

Is it possible to make it installable by brew?

@ldoench Once a working v0.2 is released I will make macchanger brew-installable.

@dennis777
Copy link

@shilch Confirmed it works 15.2! Thank you! Was the solution to rewrite it in Objective-C? Does apple not expose some API in C that is required to do the disassociation?

@shilch
Copy link
Owner

shilch commented Jan 3, 2025

Confirmed it works 15.2! Thank you! Was the solution to rewrite it in Objective-C? Does apple not expose some API in C that is required to do the disassociation?

@dennis777 Perfect! Actually, I just realized that I hardcoded the interface name for disassociation to en0. Fixed that.

The solution was to use CoreWLAN.framework, which is used by the WiFi dropdown in the status bar. The previous solution I posted reimplemented the disassociation from IO80211.framework, but that library seems to be broken as of some time ago.

CoreWLAN.framework does not expose a C API (officially). There are still traces of the Apple internal MobileWiFi.framework, but I believe it has been discontinued and I don't want to depend on it for macchanger to break again.

@dennis777
Copy link

dennis777 commented Jan 3, 2025

The solution was to use CoreWLAN.framework, which is used by the WiFi dropdown in the status bar. The previous solution I posted reimplemented the disassociation from IO80211.framework, but that library seems to be broken as of some time ago.

@shilch Makes sense Apple dropped IO80211 support. I wonder if packet sniffing and frame injections are possible using the CoreWLAN.framework? Sniffing should be as there is a built in sniffer but I don't know If frames injections are. Do you think it's possible to for example send the deauth frames?

@shilch
Copy link
Owner

shilch commented Jan 3, 2025

I wonder if packet sniffing and frame injections are possible using the CoreWLAN.framework? Sniffing should be as there is a built in sniffer but I don't know If frames injections are. Do you think it's possible to for example send the deauth frames?

After having a short look, it looks like this is possible using a /dev/bpfX device (man 4 bpf). This technique is implemented by libpcap as pcap_inject. This project appears to offer what you are looking for.

@dennis777
Copy link

dennis777 commented Jan 3, 2025

After having a short look, it looks like this is possible using a /dev/bpfX device (man 4 bpf). This technique is implemented by libpcap as pcap_inject. This project appears to offer what you are looking for.

That is the project I found years ago and am trying to rewrite but unfortunately no dice. I think apple may have disabled frame injections on a hardware level or something because the application doesn't even scan. Previously it would just crash. Also it's not possible to even compile the project now due to some signing issue. Anyway thanks just was curious if you knew more.

@artkiver
Copy link
Author

artkiver commented Jan 3, 2025

Hi everybody, sorry for the silence. I found a solution that automatically disassociates the wifi on newer macOS versions. Successfully tested on 15.2. I pushed the draft to a new branch v0.2-draft2.

Please let me know if this solution works for you and which version of macOS you are using.

git clone -b v0.2-draft2 https://github.com/shilch/macchanger
cd macchanger
make
sudo ./macchanger -r en0

I've confirmed that the v0.2-draft2 branch is working like a charm for me on 15.2 as well.

Unfortunately, last summer, someone broke into my car and stole two laptops (including a 2012 MacBook Pro I used for testing older OS versions) so I can't test more extensively at the moment.

I noticed someone else in the thread mentioned brew; and I am a MacPorts maintainer and have considered creating a Portfile for macchanger for MacPorts. Especially now that it seems to be functioning again without issue on the most -CURRENT non-beta OS release, I may look into that again (though others are welcome to contribute such things)!

I would generally dissuade security minded folks (as I assume most who are randomizing MAC addresses presumably are) from using brew entirely given that it has analytics on by default (information on how to disable them here: https://docs.brew.sh/Analytics) whereas MacPorts only has anonymous analytics gathered if users choose of their own volition to install mpstats (more information on that here: https://ports.macports.org/port/mpstats/)

Moreover, MacPorts was co-founded by jkh (Jordan Hubbard) who was a co-founder of FreeBSD and previously "Director of Engineering of Unix Technologies" at Apple for a dozen some odd years and can be thought of as the complementary /usr/ports in a BSD system for OS X/macOS whereas brew is a bit NIH syndrome. In particular, MacPorts makes an effort to automate testing on many versions of macOS/OS X (going back to at least Leopard, though some users report still having success with it on Tiger) whereas brew only tends to focus on the current and last two macOS releases.

@PianoMan7
Copy link

@shilch Confirmed it works 15.2! Thank you! Was the solution to rewrite it in Objective-C? Does apple not expose some API in C that is required to do the disassociation?

working for me too on 15.2! Thanks @shilch!

@blakegonzales
Copy link

@shilch Confirmed it works 15.2! Thank you! Was the solution to rewrite it in Objective-C? Does apple not expose some API in C that is required to do the disassociation?

working for me too on 15.2! Thanks @shilch!

works for me too in 15.2 :)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests