/* * GRUB -- GRand Unified Bootloader * Copyright (C) 2010 Free Software Foundation, Inc. * * GRUB is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * GRUB is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with GRUB. If not, see . */ #ifndef GRUB_NET_HEADER #define GRUB_NET_HEADER 1 #include #include #include #include #include #include #include typedef enum grub_link_level_protocol_id { GRUB_NET_LINK_LEVEL_PROTOCOL_ETHERNET } grub_link_level_protocol_id_t; typedef struct grub_net_link_level_address { grub_link_level_protocol_id_t type; union { grub_uint8_t mac[6]; }; } grub_net_link_level_address_t; typedef enum grub_net_interface_flags { GRUB_NET_INTERFACE_HWADDRESS_IMMUTABLE = 1, GRUB_NET_INTERFACE_ADDRESS_IMMUTABLE = 2, GRUB_NET_INTERFACE_PERMANENT = 4 } grub_net_interface_flags_t; typedef enum grub_net_card_flags { GRUB_NET_CARD_HWADDRESS_IMMUTABLE = 1, GRUB_NET_CARD_NO_MANUAL_INTERFACES = 2 } grub_net_card_flags_t; struct grub_net_card; struct grub_net_card_driver { struct grub_net_card_driver *next; char *name; grub_err_t (*init) (struct grub_net_card *dev); grub_err_t (*fini) (struct grub_net_card *dev); grub_err_t (*send) (const struct grub_net_card *dev, struct grub_net_buff *buf); grub_ssize_t (*recv) (const struct grub_net_card *dev, struct grub_net_buff *buf); }; extern struct grub_net_card_driver *grub_net_card_drivers; static inline void grub_net_card_driver_register (struct grub_net_card_driver *driver) { grub_list_push (GRUB_AS_LIST_P (&grub_net_card_drivers), GRUB_AS_LIST (driver)); } static inline void grub_net_card_driver_unregister (struct grub_net_card_driver *driver) { grub_list_remove (GRUB_AS_LIST_P (&grub_net_card_drivers), GRUB_AS_LIST (driver)); } #define FOR_NET_CARD_DRIVERS(var) for (var = grub_net_card_drivers; var; var = var->next) typedef struct grub_net_packet { struct grub_net_packet *next; struct grub_net_packet *prev; struct grub_net_packets *up; struct grub_net_buff *nb; } grub_net_packet_t; typedef struct grub_net_packets { grub_net_packet_t *first; grub_net_packet_t *last; } grub_net_packets_t; struct grub_net_card { struct grub_net_card *next; char *name; struct grub_net_card_driver *driver; grub_net_link_level_address_t default_address; grub_net_card_flags_t flags; union { struct { struct grub_efi_simple_network *efi_net; void *efi_handle; }; void *data; int data_num; }; }; struct grub_net_network_level_interface; typedef enum grub_network_level_protocol_id { GRUB_NET_NETWORK_LEVEL_PROTOCOL_PROMISC, GRUB_NET_NETWORK_LEVEL_PROTOCOL_IPV4 } grub_network_level_protocol_id_t; typedef struct grub_net_network_level_address { grub_network_level_protocol_id_t type; union { grub_uint32_t ipv4; }; } grub_net_network_level_address_t; typedef struct grub_net_network_level_netaddress { grub_network_level_protocol_id_t type; union { struct { grub_uint32_t base; int masksize; } ipv4; }; } grub_net_network_level_netaddress_t; #define FOR_PACKETS(cont,var) for (var = (cont).first; var; var = var->next) static inline grub_err_t grub_net_put_packet (grub_net_packets_t *pkts, struct grub_net_buff *nb) { struct grub_net_packet *n; n = grub_malloc (sizeof (*n)); if (!n) return grub_errno; n->nb = nb; n->next = NULL; n->prev = NULL; n->up = pkts; if (pkts->first) { pkts->last->next = n; pkts->last = n; n->prev = pkts->last; } else pkts->first = pkts->last = n; return GRUB_ERR_NONE; } static inline void grub_net_remove_packet (grub_net_packet_t *pkt) { if (pkt->prev) pkt->prev->next = pkt->next; else pkt->up->first = pkt->next; if (pkt->next) pkt->next->prev = pkt->prev; else pkt->up->last = pkt->prev; grub_free (pkt); } typedef struct grub_net_app_protocol *grub_net_app_level_t; typedef struct grub_net_socket *grub_net_socket_t; struct grub_net_app_protocol { struct grub_net_app_protocol *next; char *name; grub_err_t (*dir) (grub_device_t device, const char *path, int (*hook) (const char *filename, const struct grub_dirhook_info *info)); grub_err_t (*open) (struct grub_file *file, const char *filename); grub_err_t (*read) (grub_net_socket_t sock, struct grub_net_buff *nb); grub_err_t (*close) (struct grub_file *file); grub_err_t (*label) (grub_device_t device, char **label); }; struct grub_net_socket { struct grub_net_socket *next; int status; int in_port; int out_port; grub_net_app_level_t app; grub_net_network_level_address_t out_nla; struct grub_net_network_level_interface *inf; grub_net_packets_t packs; void *data; }; extern struct grub_net_socket *grub_net_sockets; static inline void grub_net_socket_register (grub_net_socket_t sock) { grub_list_push (GRUB_AS_LIST_P (&grub_net_sockets), GRUB_AS_LIST (sock)); } static inline void grub_net_socket_unregister (grub_net_socket_t sock) { grub_list_remove (GRUB_AS_LIST_P (&grub_net_sockets), GRUB_AS_LIST (sock)); } #define FOR_NET_SOCKETS(var) for (var = grub_net_sockets; var; var = var->next) typedef struct grub_net { char *server; grub_net_app_level_t protocol; grub_net_socket_t socket; grub_fs_t fs; } *grub_net_t; extern grub_net_t (*EXPORT_VAR (grub_net_open)) (const char *name); extern grub_err_t (*EXPORT_VAR (grub_file_net_seek)) (struct grub_file *file, grub_off_t offset); extern void (*EXPORT_VAR (grub_grubnet_fini)) (void); struct grub_net_network_level_interface { struct grub_net_network_level_interface *next; struct grub_net_network_level_interface **prev; char *name; const struct grub_net_card *card; grub_net_network_level_address_t address; grub_net_link_level_address_t hwaddress; grub_net_interface_flags_t flags; struct grub_net_bootp_packet *dhcp_ack; grub_size_t dhcp_acklen; void *data; }; struct grub_net_session; struct grub_net_session_level_protocol { void (*close) (struct grub_net_session *session); grub_ssize_t (*recv) (struct grub_net_session *session, void *buf, grub_size_t size); grub_err_t (*send) (struct grub_net_session *session, void *buf, grub_size_t size); }; struct grub_net_session { struct grub_net_session_level_protocol *protocol; void *data; }; static inline void grub_net_session_close (struct grub_net_session *session) { session->protocol->close (session); } static inline grub_err_t grub_net_session_send (struct grub_net_session *session, void *buf, grub_size_t size) { return session->protocol->send (session, buf, size); } static inline grub_ssize_t grub_net_session_recv (struct grub_net_session *session, void *buf, grub_size_t size) { return session->protocol->recv (session, buf, size); } struct grub_net_network_level_interface * grub_net_add_addr (const char *name, const struct grub_net_card *card, grub_net_network_level_address_t addr, grub_net_link_level_address_t hwaddress, grub_net_interface_flags_t flags); extern struct grub_net_network_level_interface *grub_net_network_level_interfaces; extern grub_net_app_level_t grub_net_app_level_list; #ifndef GRUB_LST_GENERATOR static inline void grub_net_app_level_register (grub_net_app_level_t proto) { grub_list_push (GRUB_AS_LIST_P (&grub_net_app_level_list), GRUB_AS_LIST (proto)); } #endif static inline void grub_net_app_level_unregister (grub_net_app_level_t proto) { grub_list_remove (GRUB_AS_LIST_P (&grub_net_app_level_list), GRUB_AS_LIST (proto)); } #define FOR_NET_APP_LEVEL(var) FOR_LIST_ELEMENTS((var), \ (grub_net_app_level_list)) extern struct grub_net_card *grub_net_cards; static inline void grub_net_card_register (struct grub_net_card *card) { grub_list_push (GRUB_AS_LIST_P (&grub_net_cards), GRUB_AS_LIST (card)); } static inline void grub_net_card_unregister (struct grub_net_card *card) { grub_list_remove (GRUB_AS_LIST_P (&grub_net_cards), GRUB_AS_LIST (card)); } #define FOR_NET_CARDS(var) for (var = grub_net_cards; var; var = var->next) struct grub_net_session * grub_net_open_tcp (char *address, grub_uint16_t port); grub_err_t grub_net_resolve_address (const char *name, grub_net_network_level_address_t *addr); grub_err_t grub_net_resolve_net_address (const char *name, grub_net_network_level_netaddress_t *addr); grub_err_t grub_net_route_address (grub_net_network_level_address_t addr, grub_net_network_level_address_t *gateway, struct grub_net_network_level_interface **interf); grub_err_t grub_net_add_route (const char *name, grub_net_network_level_netaddress_t target, struct grub_net_network_level_interface *inter); grub_err_t grub_net_add_route_gw (const char *name, grub_net_network_level_netaddress_t target, grub_net_network_level_address_t gw); #define GRUB_NET_BOOTP_MAC_ADDR_LEN 16 typedef grub_uint8_t grub_net_bootp_mac_addr_t[GRUB_NET_BOOTP_MAC_ADDR_LEN]; struct grub_net_bootp_packet { grub_uint8_t opcode; grub_uint8_t hw_type; /* hardware type. */ grub_uint8_t hw_len; /* hardware addr len. */ grub_uint8_t gate_hops; /* zero it. */ grub_uint32_t ident; /* random number chosen by client. */ grub_uint16_t seconds; /* seconds since did initial bootstrap. */ grub_uint16_t flags; grub_uint32_t client_ip; grub_uint32_t your_ip; grub_uint32_t server_ip; grub_uint32_t gateway_ip; grub_net_bootp_mac_addr_t mac_addr; char server_name[64]; char boot_file[128]; grub_uint8_t vendor[0]; } __attribute__ ((packed)); #define GRUB_NET_BOOTP_RFC1048_MAGIC_0 0x63 #define GRUB_NET_BOOTP_RFC1048_MAGIC_1 0x82 #define GRUB_NET_BOOTP_RFC1048_MAGIC_2 0x53 #define GRUB_NET_BOOTP_RFC1048_MAGIC_3 0x63 struct grub_net_network_level_interface * grub_net_configure_by_dhcp_ack (const char *name, const struct grub_net_card *card, grub_net_interface_flags_t flags, const struct grub_net_bootp_packet *bp, grub_size_t size, int is_def, char **device, char **path); void grub_net_process_dhcp (struct grub_net_buff *nb, const struct grub_net_card *card); int grub_net_hwaddr_cmp (const grub_net_link_level_address_t *a, const grub_net_link_level_address_t *b); /* Currently suppoerted adresses: IPv4: XXX.XXX.XXX.XXX */ #define GRUB_NET_MAX_STR_ADDR_LEN sizeof ("XXX.XXX.XXX.XXX") void grub_net_addr_to_str (const grub_net_network_level_address_t *target, char *buf); extern struct grub_net_network_level_interface *grub_net_network_level_interfaces; #define FOR_NET_NETWORK_LEVEL_INTERFACES(var) for (var = grub_net_network_level_interfaces; var; var = var->next) grub_err_t grub_net_send_link_layer (struct grub_net_network_level_interface *inf, struct grub_net_buff *nb, grub_net_link_level_address_t *target); typedef int (*grub_net_packet_handler_t) (struct grub_net_buff *nb, struct grub_net_network_level_interface *inf); grub_err_t grub_net_recv_link_layer (struct grub_net_network_level_interface *inf, grub_net_packet_handler_t handler); grub_err_t grub_net_recv_ip_packets (struct grub_net_buff *nb, const struct grub_net_card *card, const grub_net_link_level_address_t *hwaddress); grub_err_t grub_net_send_ip_packet (struct grub_net_network_level_interface *inf, const grub_net_network_level_address_t *target, struct grub_net_buff *nb); #define FOR_NET_NL_PACKETS(inf, var) FOR_PACKETS(inf->nl_pending, var) void grub_net_poll_cards (unsigned time); #endif /* ! GRUB_NET_HEADER */